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ec 于 
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程序 员 在 求职 时 , 经 常会 被 问 到 有 关 开 发 经 验 的 问题 , 例如 “从 事 了 多 少年 的 开发 "”、“ 参 
与 了 哪些 大 型 项 目 ”等 。 为 何 面试 官 对 项 目 经 验 丰 富 的 求职 者 青睐 有 加 ? 究 其 根源 在 于 程序 开 
发 是 一 门 实践 性 非常 强 的 学 科 ， 正 所 谓 “ 纸 上 得 来 终 觉 浅 ， 绝 知 此 事 要 躬 行 ”， 编 程 经 验 是 程 
序 开发 者 在 实践 中 积累 的 技术 资源 库 ， 其 中 包括 各 种 习惯 用 法 、 知 识 理论 、 代 码 片段 、 应 用 技 
巧 等 。 而 编程 高 手 一 般 也 是 站 在 前 人 的 肩膀 上 ， 分 析 别 人 的 代码 而 有 所 领悟 ， 或 者 借鉴 他 人 的 
经 验 技巧 形成 自己 的 技能 ， 在 认识 错误 与 解决 问题 中 进步 。 这 也 是 每 个 编程 者 的 必由之路 。 

本 书 汇集 了 一 线 开 发 人 员 常 见 的 300 个 PHP 编程 疑难 问题 ， 每 个 问题 都 给 出 了 详细 的 解 
答 与 点 评 ， 图 文 并 茂 ， 难 易 并 举 ， 通 过 本 书 的 学 习 ， 相 信 读 者 可 以 尽 享 前 人 的 开发 经 验 ， 迅 速 
提升 个 人 的 开发 能 力 。 


本 书 内 容 


本 书 以 PHP 基础 知识 结构 为 框架 ， 给 出 了 每 部 分 知识 中 可 能 遇 到 的 疑难 问题 或 开发 技巧 。 
全 书 共 分 为 16 章 ， 具体 包 括 : PHP 开发 规范 与 入 门 要 点 、PHP 基础 、Web 技术 、 数 据 库 操作 、 
日 期 和 时 间 处 理 、 图 形 图 像 处 理 、 文件 与 目录 操作 、 面 向 对 象 的 程序 设计 、 数据 库 抽象 层 、Ajax 
技术 、XML 技术 、PDF 技术 、PHP 与 多 媒体 技术 、Smarty 模板 、ThinkPHP 框架 和 PHP 项 目 
开发 技巧 。 

为 了 更 清晰 地 图 述 问题 和 给 出 问题 的 解决 方案 ， 本 书 设置 了 以 下 栏目 。 

回 问题 阐述 

对 问题 进行 详细 阐述 ， 对 复杂 问题 给 出 问题 产生 的 条 件 ， 并 对 问题 中 需要 解决 的 部 分 做 出 
说 明 。 

加 专家 解答 

根据 具体 问题 进行 具体 分 析 ， 给 出 问题 的 正确 答案 及 解 题 思 路 ， 并 提供 该 问题 涉及 的 技术 
知识 。 

加 专家 点 评 

对 问题 及 解答 做 出 总 结 ， 为 以 后 再 出 现 类 似 问题 提供 解决 思路 ， 并 给 出 此 问题 需要 注意 的 
事项 和 对 该 问题 的 一 些 拓展 。 


本 书 特 色 


回 ”贴近 应 用 
本 书 精 选 的 疑难 问题 都 是 在 实际 项 目 开 发 中 经 常 遇 到 的 问题 , 可 以 解决 读者 在 编程 中 可 能 


人 Pp 六 须知 道 的 5300 个 问题 


| 遇 到 的 疑惑 。 


| 能 力 。 


回 ”横向 链接 
本 书 知识 框架 与 《PHP 快速 入 门 及 项 目 实战 》 一 书 相对 应 ， 读 者 可 以 在 使 用 《PHP 快速 入 


| 门 及 项 目 实战 》 一 书 进行 基础 学 习 之 后 ， 再 使 用 本 书 丰富 并 提高 技能 。 


回 ”解析 透彻 

本 书 对 每 个 问题 的 相关 知识 进行 细致 地 讲解 ， 并 进行 知识 拓展 ， 使 读者 不 仅 知 其 然 ， 而 且 
知 其 所 以 然 。 

回 授信 以 渔 
本 书 在 讲解 技术 的 同时 还 注重 对 读者 能 力 的 培养 ， 帮 助 读者 提高 分 析 问 题 与 解决 问题 的 


本 书 配套 资源 


本 书 提供 了 内 容 丰富 的 配套 资源 包括 : 源 程序 、 素 材 ， 以 及 模块 库 、 案 例 库 、 题 库 、 素 材 


| 库 等 多 项 辅助 内 容 ， 读 者 朋友 可 以 通过 如 下 方式 获取 。 


第 1 种 方式 : 
(1) 登录 www.tup.com.cn， 在 网 页 右上 角 的 搜索 文本 框 中 输入 本 书 书 名 注意 区 分 大 小 


| 写 和 留 出 空格 ), 或 者 输入 本 书 关键 字 , 或 者 输入 本 书 ISBN 号 (注意 去 掉 ISBN 号 间隔 线 “-”)， 


| 写 
| 单 击 “搜索 ”按钮 。 


(2) 找到 本 书后 单 击 超 链接 ， 在 该 书 的 网 页 下 侧 单 击 “ 网 络 资源 ” 超 链接 ， 即 可 下 载 。 
第 2 种 方式 : 
访问 本 书 的 新 浪 微 博 PHPbook， 找 到 配套 资源 的 链接 地 址 进行 下 载 。 


| 读者 人 群 


本 书 非 常 适合 以 下 人 员 阅 读 : 


回 从 事 PHP 语言 编程 行业 的 开发 人 员 

回 有 一 定语 言 基础 ， 想 进一步 提高 技能 的 人 员 

回 大 中 专 院 校 的 老师 和 学 生 

回 即将 走 上 工作 岗位 的 大 学 毕业 生 

回 ”相关 培训 机 构 的 老师 和 学 员 

回 PHP 编程 爱好 者 
读者 服务 & 本 书 勘误 


读者 在 使 用 本 书 过 程 中 遇 到 的 所 有 问题 ， 均 可 通过 以 下 方式 联系 我 们 。 

1. 新 浪 微 博 : PHPbook。 

及 时 发 布 读者 答疑 、 本 书 勘 误 、 配 套 资 料 更 新 等 内 容 。 

2. 腾讯 QQ: 4006751066。 

3. 登录 网 站 : www.mingribook.com， 在 论坛 、 勘 误 发 布 、 读 者 纠 错 、 技 术 支持 、 读 者 
之 家 等 栏目 中 的 相关 模块 中 提问 、 留 言 或 查看 。 


本 书 作者 
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问题 1 3 如 何 解决 Apache 服务 器 端口 
冲突 ? 

问题 14 如 何 配 置 Apache 虚拟 主机 ? 
问题 13 ”如何 动态 加 载 扩展 库 ? 

问题 16 ”register globals (注册 为 全 局 变 
量 ) 的 开启 与 关闭 

问题 17 为 什么 要 设置 时 区 ? 


如 何在 Windows 下 配置 PHP 开 


如 何在 Linux 下 配置 PHP 开发 


问题 18 ”如 何在 Dreamweaver 中 创建 站 点 ? 
问题 19 如何 使 用 Dreamweaver 开发 PHP 
程序 ? 

问题 20 ”如 何 使 用 Dreamweaver 识别 phtml 
的 模板 文件 ? 

问题 21 如 何在 Windows 下 安装 和 配置 
phpMyAdmin? 

问题 22 如 何在 Linux 下 安装 和 配置 
phpMyAdmin? 

问题 23 如何 应 用 phpMyAdmin 修改 数 
据 库 用 户 名 和 密码 ? 

问题 24 如 何 应 用 phpMyAdmin 创建 数 
据 库 与 数据 表 ? 

问题 25 如何 应用 phpMyAdmin 导入 和 
导出 MySQL 数据 库 脚本 ? 

问题 26 如何 打开 magic quotes_gpc 来 
防止 SQL 注入 ? 

问题 27 如 何 对 错误 提示 信息 进行 

控制 ? 

问题 28 如 何 记录 错误 日 志 ? 

问题 29 如 何 解 决 使 用 MySQL 函数 导致 
Apache(IIS) 崩 涡 的 问题 ? 

问题 30 HTTP Apache 错误 代码 都 有 哪 
些 ? 分 别 代 表 什 么 含义 ? 
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问题 ] 如 何在 Windows 下 配置 PHP 开发 环境 ? 


问题 讲述 
PHP 高 效 、 稳 定 的 运行 依赖 于 服务 器 的 编译 和 执行 ， 那 么 如 何在 微软 的 Windows 操作 系 


| 统 中 架设 安全 、 可 靠 的 PHP 运行 环境 呢 ? 
| 专家 解答 


1. Windows 下 Apache 的 安装 配置 
Apache 是 全 世界 范围 内 使 用 最 广 的 Web 服务 软件 。 超 过 50% 的 网 站 都 在 使 用 Apache， 它 


以 高 效 、 稳 定 、 安 全 、 免 费 〈 最 重要 的 一 点 ) 的 优点 而 成 为 最 受 欢 迎 的 服务 器 软件 。 


本 节 主 要 介绍 如 何在 Windows 操作 系统 中 安装 和 配置 Apache 服务 器 。 在 安装 Apache 服 


| 务 器 前 ， 应 到 其 官方 网 站 http://www.apache.org 下 载 Apache 的 安装 程序 。 


在 Windows 下 实现 Apache 的 安装 的 操作 步骤 如 下 。 
(1) 下 载 Apache 的 安装 包 “apache 2.0.59-win32-x86-no_sslmsi” 后 双击 该 安装 包 ， 打 开 


| 如 图 1.1 所 示 的 Apache 的 安装 窗口 。 


(2) 在 图 1.1 所 示 的 窗口 中 单 击 Next 按钮 ， 打 开 如 图 1.2 所 示 的 Apache 许可 协议 窗口 ， 


| 选中 图 中 的 Iaccept the terms in the license agreement 单 选 按钮 ， 然 后 单 击 Next 按钮 。 


| 中 ri x 
‘Welcome to the installation Wizard for 
Apache HTTP Server 2.0.59 Messe resd the plowr teenee aresment carefuly. 
The Instalation Waard wl nctall Apache HTTP Server 20.59 a 
on your Compcker ocorenue, dei Mest ee lene 习 
http:ywww .apache.orgmicenses/ 
[TERMS AND CONDmIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 
1 Dern 
WARNING: Ths program 5 protected by copymgpt low and 可 > yon,an 
maoral ae me 选择 接受 许可 协议 "ni 
一 
Re nmolgheld 
baik [CE _cameel <Back Next > anesl 


图 1.1 Apache 的 安装 窗口 图 1.2 许可 协议 窗口 
(3) 打开 如 图 1.3 所 示 的 HTTP 服务 窗口 ， 单 击 Next 按钮 ， 打 开 如 图 1.4 所 示 的 输入 服 


| 务 器 信息 窗口 。 输 入 服务 器 的 相关 信息 ， 如 网 络 域名 、 服 务 器 名 和 管理 员 邮 箱 等 ， 这 里 可 以 根 
| 据 用 户 的 实际 情况 进行 输入 。 在 下 方 的 单 选 按 钮 组 中 ， 如 果 选 择 第 一 项 则 可 以 对 任何 用 户 开放 
| Apache 服务 ， 同 时 设置 服务 器 的 侦 听 端口 为 “80”; 如果 选 择 第 二 项 则 只 有 本 地 用 户 可 以 连接 
| 和 使 用 Apache 服务 。 


1 
注意: 


如 果 选 择 的 是 第 一 项 , 且 同时 安装 了 IIS, 那 就 必须 修改 IIS 的 默认 端口 ,否则 将 导致 Apache 


| 无 法 正常 工作 。 更 改 IIS 的 默认 侦 听 端 口 80， 可 以 在 TIS 的 管理 器 中 进行 设置 ， 或 者 停止 IIS 
| 的 服务 也 可 以 。 
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Read This First 
Read this Before Running Apache on Wndows, 
Apache HTTP Sever i 
4 


Foundaton th developers aimto colaboratvely deveppand maintan 3 robust 
|eommerelargrada, standards-basad semer win reey avalable sourre r00e 


[rhe Latest vorsion 
[Detalls ofthe latest version can be fpund on the Apa:he HTTP server prcject page Instal Apache HITF Sever 20 progans and shorteuts for, 


for MW Users, on Port go 0 5ervice ~ Recommonded, 
WT UE Cre Ue FT EOS0 Wren TateT amd, 


ce [ae cme | 


图 1.3 HTTP 服务 窗口 图 1.4 输入 服务 器 信息 窗口 
(4) 设置 完成 后 ， 单 击 图 1.4 中 的 Next 按钮 ， 打 开 如 图 1.5 所 示 的 选择 安装 方式 窗口 。 
在 该 窗口 中 有 “Typical”( 典 型 安装 ) 和 “Custom”( 自 定义 安装 ) 两 种 安装 方式 。 通 常 ， 用 户 
都 选择 典型 安装 方式 ， 单 击 Next 按钮 。 | 
(5) 打开 如 图 1.6 所 示 的 设置 安装 的 路 径 窗 口 ， 在 其 中 可 以 选择 安装 的 路 径 ， 单 击 Change | 


按钮 。 


Setup Type Destination Folder 
choose the setup type that best sults your needs, Ch chengs to notal to a difierent foder 


Ploae oolect 3 aokup typo. -Instal Apache HTTP Server 2.0to the foder: 


四 
ph 设置 Apache 的 安装 路 径 
选择 典型 安装 


Custom 
choose which program featuras you wankinstaled and where they 
beinsaled Reccmnerded for advanced users, 


Jnstalsilad Trayalf 


ms Ce ee | IC ca 


图 1.5 选择 安装 方式 图 1.6 设置 安装 的 路 径 


(6) 打开 如 图 1.7 所 示 的 修改 文件 的 安装 路 径 窗口 ， 在 其 中 对 安装 的 路 径 进行 修改 ， 然 后 | 
单 击 OK 按钮 。 | 
(7) 打开 如 图 1.8 所 示 的 准备 安装 窗口 ， 单 击 其 中 的 Install 按钮 。 


昌 Apache =- i 可 sr 2 FE | 
Change Curent Destination Folder Ready te Install the program 
Browse to the destinaton foder, The wizsrd is raady te begininstalaion 


Lookn: Cick Install to begin the installation. 
Apache Grour ”| FF you want to review or change any of your installation settings, click Back, Cick Cancel to 
Eh mate emp EW et 


me [a] ere | 
图 1.7 修改 安装 的 路 径 图 1.8 准备 安装 


(8) 打开 如 图 1.9 所 示 的 开始 安装 窗口 ， 在 其 中 所 有 文件 复制 完成 后 ，Apache 安装 完成 ， 
打开 如 图 1.10 所 示 的 窗口 ， 单 击 Finish 按钮 完成 Apache 的 安装 。 


Tats: 


Boot whi te Irstaloton Weard hstale apache HTTP Sorver 
0.59. This may take several minutes, 


Copyingnew files 


a 


Istalshield 


Installation Wizard Completed 


The Petalakon Wizard has sucressfuly retaled Apache HTTP 
Server 20.59, CickFinsh to edt the Nizard, 


图 1.9 复制 文件 图 1.10 安装 完成 


(9) 接 下 来 测试 Apache 服务 是 否 安装 成 功 。 选 择 “ 开 始 ” 一 
的 菜单 中 能 够 看 到 Apache 服务 器 相关 操作 列表 ， 同 时 如 果 在 系统 托盘 中 有 一 个 师 图 标 ， 则 表 
示 Apache 服务 已 经 启动 。 


(10) 单 


“所 有 程序 ”命令 ， 在 弹出 


辆 图 标 ， 打 开 如 图 1.11 所 示 的 窗口 。 在 该 窗口 中 ， 选 择 Stop 选项 表示 停 让 


Apache 服务 器 ， 选择 Start 选项 表示 启动 Apache 服务 器 〈 当 Apache 服务 器 的 当前 状态 为 停止 
时 ， 该 功能 可 用 ); 选择 Restart 选项 表示 重新 启动 Apache 服务 器 。 双击 短 图 标 后， 打开 如 
图 1.12 所 示 的 窗口 ， 可 以 进行 启动 和 停止 服务 器 的 各 种 操作 。 


图 1.11 


启动 Apache 服务 器 


[ah £3 inc) 


图 1.12 Apache 服务 器 的 监视 器 


(11) 在 浏览 器 中 输入 “http://localhost/” 或 “http://127.0.0.1/”， 如 果 能 够 浏览 到 如 图 1.13 
所 示 的 窗口 ， 则 说 明 Apache 服务 器 安装 成 功 。 
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这 不 是 你 想 看 见 的 页 面 吧 ? 
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图 1.13 测试 Apache 服务 器 是 否 安 装 成 功 
“4. 
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Apache 服务 器 安装 成 功 后 ， 接 下 来 对 Apache 服务 器 进行 配置 ， 以 便 Apache 服务 器 能 
识别 PHP 文件 。 

具体 步骤 如 下 : | 

(1) 配置 Apache 服务 器 主要 是 在 Apache 安装 目录 下 的 conf 子 目录 中 的 htpdconf 文件 | 

中 进行 的 ， 找 到 该 文件 并 用 记事 本 等 文本 编辑 器 打开 。 | 

(2) 定位 到 “DocumentRoot” 一 行 ， 可 以 将 该 路 径 修改 为 合适 的 路 径 。 例 如 ， 设 置 为 | 

“DocumentRoot"C:\Apache\Apache2\webpage"” 表 示 Apache 服务 器 的 虚拟 目录 为 “C:\Apache\ | 

Apache2vwebpage ”。 | 

(3) 将 光标 定位 到 下 面 这 行 代码 : DirectoryIndex index html index htmlvar， 在 其 后 面 添加 | 

一 个 PHP 默认 页 ， 通 常 是 indexphp， 表 示 当 访问 该 服务 器 时 如 果 未 指定 要 访问 的 PHP 文件 ， | 
则 默认 访问 index.php 文件 ， 更 改 后 的 代码 如 下 。 

DirectoryIndex index.html index.htmlvar index.php | 

(4) 为 了 使 Apache 能 够 识别 PHP 的 扩展 名 , 搜索 并 定位 到 httpd.conf 文件 的 下 面 这 个 部 分 。 | 


<Directory "C:/Apache/Apache2/cgi-bin"> 
AllowOverride None 
Options None 
Order allow,deny 
Allow from all 
</Directory> 


在 后 面 添 加 如 下 两 行 代码 。 


AddType application/x-httpd-php .php .phtml .php3 .php4 
AddType application/x-httpd-php-source .phps 

(5) 为 了 能 够 使 用 模块 功能 ， 模 块 通常 以 DSO 方式 构建 ， 需 要 定位 到 如 下 这 行 代码 : 
#0adModule ssl module modules/mod ssl.so 


使 得 能 够 在 使 用 前 获得 指令 的 功能 ， 然 后 以 module 方式 加 载 PHP， 指 向 PHP 5.0 目录 下 | 
的 php5apache2.dll 文件 ， 添 加 如 下 代码 。 | 
LoadModule php5_module ¢:\php5\php5apache2.dll 

2. Windows 下 MySQL 的 安装 配置 | 
PHP 能 够 支持 市 面 上 的 大 多 数 数据 库 , 例如 MySQL、 Access、 SQLServer、 Oracle 等 ,MySQL | 
数据 库 与 PHP 是 程序 开发 人 员 公认 的 黄金 搭档 , 这 不 仅 因为 MySQL 是 完全 网 络 化 的 跨 平台 关 | 
系 型 数据 库 系 统 ， 还 因为 MySQL 是 具有 客户 机 /服务 器 体系 结构 的 分 布 式 数据 库 管 理 系统 ， 该 | 
数据 库 具 有 功能 性 强 、 使 用 简捷 、 管 理 方便 、 运 行 速度 快 、 版 本 升级 快 、 安 全 性 高 等 优点 ， 而 | 
且 是 完全 免费 的 ， 从 其 官方 网 站 http://www.MySQL.com 即 可 免费 下 载 最 新 版 本 的 安装 包 。 | 
在 Windows 下 实现 MySQL 的 安装 配置 的 操作 步骤 如 下 。 | 

(1) 如 果 下 载 的 是 Windows Essentials (x86)， 例 如 MySQL-4.1.11-essential-win.exe， 则 | 

直接 双击 该 文件 安装 即 可 。 安 装 完毕 后 ， 选 择 “ 开 始 ” 一 “运行 ”命令 ， 在 弹出 对 话 框 的 文本 | 
框 中 输入 “cmd”， 进 入 “命令 提示 符 ” 界 面 , 输入 “MySQLd-nt.exe”， 即 可 启动 MySQL 服务 。 | 
如 果 还 想 详细 地 配置 MySQL, 则 需要 进入 “C\MySQDbin "目录, 运行 MySQLInstanceConfig exe， | 
然后 按 提 示 操 作 即 可 。 | 
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(2) 如 果 下 载 的 是 Windows (x86)， 例 如 MySQL-essential-5.0.24-win32.msi， 则 解压 该 文 

件 后 双击 setup.exe 安装 程序 即 可 逐步 完成 MySQL 的 安装 。 
(3) 如 果 下 载 的 是 Without noinstall (unzip in C:\)， 则 直接 将 该 文件 解压 到 指定 的 安装 目 
录 下 。 选 择 “ 开 始 ” 一 “运行 ”命令 ， 在 弹出 对 话 框 的 文本 框 中 输入 “cmd”， 进 入 “命令 提示 
符 ” 界 面 ， 输 入 “MySQLd-nt.exe” 命 令 ， 即 可 启动 MySQL 服务 器 。 

3. Windows 下 PHP 的 安装 配置 

架设 基于 PHP 的 Web 服务 器 ,安装 PHP 是 必须 的 。 由 于 PHP 的 代码 是 公开 的 ， 所 以 其 升 
级 速度 较 快 。 安 装 PHP 之 前 应 从 官方 网 站 http://www.php.net/ 下 载 最 新 版 本 的 PHP 安装 程序 。 
下 面 以 php-5.1.4-Win32.zip 为 例 介绍 PHP 的 安装 配置 方法 。 

在 Windows 下 实现 PHP 的 安装 配置 的 操作 步骤 如 下 。 


(1) 成 功 下 载 PHP 的 安装 包 后 ， 首 先 应 对 其 进行 解压 ， 如 解压 到 C:\php5\ 目 录 下 ， 如 
图 1.14 所 示 。 
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图 1.14 PHP 包装 包 解 讨 后 的 文件 


(2) 将 Ci\php5\ 目 录 下 的 所 有 .dll 文件 复制 到 C:\Windows\system32\ (如 果 是 Windows 
2000 操作 系统 ， 则 为 C\WINNT\system32\) 目录 下 。 
(3) 将 Ci\php 引 目录 下 的 php.ini-dist 文件 复制 到 C:\Windows\〈 如 果 是 Windows 2000 操 
作 系 统 ， 则 为 C\WINNT\) 目录 下 ， 然 后 将 php.ini-dist 重 命名 为 php.ini， 并 用 记事 本 打开 该 文 
件 进行 编辑 ， 具 体 编辑 方式 如 下 。 
首先 ， 定 位 到 该 文件 extension_dir="\" 所 在 行 ， 将 路 径 改 为 "C:\php5\ext"。 代 码 如 下 。 
extension_dir="C:\php5\ext" 
然后 ， 查 找 定位 Windows Extensions， 将 下 面 两 行 代码 前 面 的 注释 ( 即 分 号 标识 符 ) 去掉 。 


; extension=php_gd2.dll 
; extension=php MySQL.dll 
这 样 PHP 就 可 以 支持 GD2 函数 库 和 MySQL 相关 函数 了 ， 如 果 想 让 PHP 支持 其 他 功能 和 
数 ， 同 样 需要 将 与 这 些 操作 函数 相关 的 .dll 文件 前 的 注释 去 掉 。 
注意 : 
以 上 为 PHP 的 常规 配置 ， 在 PHP 的 官方 论坛 上 可 以 查看 到 更 多 的 详细 配置 。 由 于 PHP 的 
版 本 有 多 种 ， 所 以 php.ini 的 配置 也 存在 差别 ， 具 体 使 用 时 应 根据 实际 情况 而 定 。 
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(4) 当 PHP 配置 完成 后 需要 重新 启动 Apache 服务 器 ， 然 后 编写 testphp 文件 并 输入 如 下 | 
代码 。 | 


<?php 
hpinfo(); | # 
到 pinfo() 食 办 
将 文件 保存 到 Apache 安装 目录 下 的 webpage 子 目录 下 , 然后 在 浏览 器 中 输入 “http://127.0.0.1/ 
testphp”， 如 果 显 示 如 图 1.15 所 示 的 页 面 则 说 明 PHP 配置 成 功 。 | 
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图 1.15 配置 成 功 


技巧 : ! 
安装 文件 的 路 径 也 要 遵循 一 定 的 客观 原则 , 为 了 避免 在 Windows 和 Linux 间 移 植 程序 时 带 | 
来 的 不 便 ， 当 选择 D:\usr\local\php 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹配 。 建 议 最 好 不 要 | 
选择 中 间 有 空格 的 目录 , 如 E:\program Files\PHP, 这 样 做 会 导致 一 些 未 知 错误 或 崩溃 情况 发 生 。 | 


问题 2 如 何在 Linux 下 配置 PHP 开发 环境 ? 


问题 阐述 


在 Linux 下 搭建 PHP 开发 环境 比 Windows 下 要 复杂 很 多 。 除 了 需要 安装 Apache、PHP 等 
软件 外 ， 还 要 安装 一 些 相关 工具 ， 设 置 必 要 参数 。 另 外 ， 如 果 要 使 用 PHP 扩展 库 ， 还 要 进行 | 
编译 。 在 Linux 下 配置 PHP 开发 环境 的 具体 的 过 程 是 什么 ? | 
专家 解答 | 

首先 打开 Linux 终端 (Linux 下 几乎 所 有 的 软件 都 需要 在 终端 下 安装 )。 在 RedHat 9 中 选 | 
择 * 诗 菜单 ” a “系统 工具 ” Pe “终端 ” 命令 。 ! 
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多 % 
1. Linux 下 安装 Apache 
首先 到 官方 网 站 http://www.apache.org 下 载 Linux 下 “httpd2.2.8.targz” 的 压缩 包 。 
在 Linux 下 安装 和 配置 Apache 服务 器 的 操作 步骤 如 下 。 

(1) 进入 Apache 安装 文件 的 目录 下 ， 如 /usr/local/work。 
cd /usr/local/work/ 


(2) 解压 安装 包 ， 完 成 后 进入 httpd2.2.8 目录 中 。 
! 


tar xfz httpd2.2.8.tar.gz 
cd httd2.2.8 


(3) 建立 makefile， 将 Apache 服务 器 安装 到 userlocal/Apache2 目录 下 。 
.configure -prefix=/usrlocal/Apache2 -enable-module=so 
(4) 编译 文件 。 
make 
(5) 开始 安装 。 
make install 
(6) 安装 完成 后 ， 在 终端 窗口 中 输入 如 下 命令 启动 或 重启 Apache 服务 。 
/usrlocal/Apache2/bin/Apachectl start 
/usr/local/Apache2/bin/Apachect] restart 
| (7) 打开 Mozilla 浏览 器 ， 在 地 址 栏 中 输入 “http:Wlocalhosty ”， 回 车 后 如 果 看 到 如 图 1.16 
| 所 示 的 页 面 ， 则 说 明 Apache 服务 器 安装 成 功 。 
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您 能 看 见 这 个 页 面 ， 说 明 Apache web seryer 已 经 安装 成 功 。 您 可 以 在 这 个 目录 中 增加 内 
容 ， 或 者 把 这 个 页 面 替换 掉 。 
这 不 是 你 想 看 见 的 页 面 吧 ? 


ts 是 因为 网 站 管理 员 改 挛 了 这 个 站 点 的 识 置 。 如 时 有 乡 问 ， 请 冰 询 维 
护 此 站 点 的 人 Apache 软 件 基金 会 . 入 和 的 各 后 全 用 的 网 且 各 和 作 的 省 ， 不 负 
Re 也 无 法 为 您 解决 设置 上 的 问题 


Apache 文档 已 经 包含 在 此 发 行 版 中 。 
您 可 以 在 使 用 Apache 的 同 站 服务 器 上 ， 自 由 地 使 用 下 面 的 图 片 。 感 谢 使 用 Apache! 
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图 1.16 Linux 下 的 Apache 服务 器 安装 


2. Linux 下 安装 PHP 
在 安装 PHP5 之 前 ， 需 要 首先 查看 libxml 的 版 本 号 。 如 果 libxml 版 本 号 小 于 2.5.10， 则 需 
| 要 先 安装 libxml 的 高 版 本 。 安 装 libxml 和 PHP5 的 步骤 如 下 如果 不 需要 安装 libxml， 则 直接 
| 跳 到 PHP5 的 安装 步骤 即 可 )。 
(1) 将 libxml 和 PHPS 复制 到 /usr/local/work/ 目 录 下 ， 并 进入 该 目录 中 。 
cd /ust/local/work 
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(2) 分 别 将 libxml2 和 PHP 解压 。 


tar xfz libxml2-2.6.62.tar.gz 
tar xfz PHP-5.2.5.tar.gz 


(3) 进入 libxml 目录 ， 建 立 makfile， 将 libxml 安装 到 /usrlocallibxml2 目录 下 。 


cd libxml2-2.6.62 
-configure -prefix=/usrlocallibxml2 


(4) 编译 文件 。 | 
make | 
(5) 开始 安装 。 | 
make install | 


(6) libxml2 安装 完毕 ， 开 始 安装 PHP5， 进 入 php-5.2.5 目录 下 。 
cd php-5.2.5 | 
(7) 建立 makefile。 | 


./configure —with-apxs2=/usr/local/Apache2/bin/apxs \ 
--with-mysql=/usr/local/mysql \ 
--with-libxml-dir=/usr/local/libxml2 \ 
--enable-sockets \ 

--enable-mbstring \ 

--enable-soap 


(8) 开始 编译 。 
make 
(9) 开始 安装 。 
make install | 
(10) 复制 php.ini-dist 或 php.ini-recommended 到 /usrlocallib 目录 ， 并 命名 为 php.ini。 | 
cp php.ini-dist /usr/local/lib/php.ini | 
《11) 更 改 httpd.conf 文件 相关 设置 ， 该 文件 位 于 asrlocal/Apache2/conf 目录 中 。 找 到 该 | 
文件 中 的 如 下 指令 行 : | 
AddType application/x-gzip .gz .tgz | 
在 该 指令 后 加 入 如 下 指令 : | 
AddType application/x-httpd-php .php | 
(12) 重新 启动 Apache， 并 在 Apache 主 目 录 下 建立 文件 phpinfo.php。 | 
<2php | 
phpinfoO; | 
?> 
(13) 在 Mozilla 浏览 器 中 输入 “http://localhost/phpinfo.php”， 如 果 出 现 如 图 1.17 所 示 的 
页 面 ， 则 表明 PHP 安装 成 功 。 
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图 1.17 phpinfo 信息 


3，Linux 下 安装 MySQL 
在 Linux 下 安装 MySQL， 需 要 到 官方 网 站 http://www.MySQL.com 下 载 Linux 下 MySQL 


的 安装 包 。Linux 下 MySQL 的 安装 配置 步骤 如 下 : 


(1) 将 下 载 的 mysql-5.1.56.tar gz 复制 到 /softs (此 为 自 定义 文件 夹 ， 用 于 存储 压缩 包 ) 下 。 
(2) 添加 用 户 和 用 户 群 组 。 


groupadd mysql 
useradd -g mysql mysql 


(3) 进入 存放 安装 文件 的 文件 夹 softs (自行 创建 的 一 个 存放 安装 文件 的 文件 夹 )。 
cd /softs 
(4) 解压 数据 库 文件 。 
tar -vzxf mysql-5.1.56.tar.gz 
(5) 进入 解压 后 的 mysql 文件 夹 。 
cd mysql-5.1.56 
(6) 安装 配置 文件 。 
./configure --prefix=/usr/local/mysql 
(7) 编译 文件 。 
make 
(8) 安装 编译 文件 。 
make install 
(9) 进入 mysql 安装 目录 。 
cd /usr/local/mysql 
(10) 提供 新 的 文件 所 有 者 。 
chown -R mysql . 
。10 。 
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(11) 提供 新 的 文件 群 组 所 有 者 。 
chgrp -R mysql . 
(12) 初始 化 数据 目录 。 


cd /usr/local/mysql/bin 
/mysql_install_ db --user=mysql 


(13) 启动 mysql。 
-mysqld safe & 
(14) 设置 mysql 数据 库 密码 。 
/mysqladmin -u root password 111 
(15) 登录 mysql 数据 库 。 


/mysql -u root -p | 
(16) 关闭 数据 库 〈 这 个 很 特殊 ， 要 注意 进程 问题 )。 | 
cd /usr/local/mysql/bin | 


-mysqladmin -u root shutdown -p 
(17) 创建 新 的 配置 文件 。 | 


cd /usr/local/mysql/share/mysql 
cp my-huge.cnf /etc/my.cnf 
chmod 777 /etc/my.cnf 


注意 : 
在 创建 新 的 配置 文件 时 ， 不 同 的 内 存 需 要 使 用 不 同 的 配置 文件 来 创建 新 的 配置 文件 。 


my-small.cnf 。”// 适 用 于 小 于 64MB 的 服务 器 
my-medium.cnf /适用 于 物理 内 存在 28MB 到 64MB, 或 者 物理 内 存在 128MB 以 上 , 但 要 运行 | 

其 他 程序 的 服务 器 | 
my-large.cnf 。”// 适 用 于 物理 内 存在 512MB 以 上 的 专用 于 数据 库 的 机 器 | 
my-huge.cnf // 适 用 于 物理 内 存 1GB 到 2GB 的 专用 于 数据 库 的 机 器 | 
my-innodb-heavy-4Gcnf /适用 于 物理 内 存在 4GB 及 以 上 专用 于 数据 库 的 机 器 ， 且 需要 复杂 查询 | 


(18) 修改 数据 库 字符 集 ， 在 新 创建 的 /etc/my.cnf 中 进行 添加 ， 注 意 其 添加 的 位 置 。 
在 [client] 下 添加 : 
default-character-set=utf8 
在 [mysqld] 下 添加 : 


default-character-set=utf8 | 
init_connect='SET NAMES utf8" 


设置 完毕 ， 登 录 数 据 库 ， 使 用 下 面 的 命令 查询 字符 集 的 状态 。 | 
status | 
(19) 对 数据 库 进行 备份 和 还 原 。 
mysqldump -u root -p root -R 数据 库 名 >c:/data.sql (备份 ，c:/ 为 自 定义 的 一 个 存储 位 置 ，data.sql 
oii 


人 Pp 六 须知 道 的 300 个 问题 


Ce 
Eh 
为 自 定义 的 存储 文件 名 称 ) 
mysql -uroot -proot -D 数据 库 名 <e:/data.sql (还 原 ，c:/data.sql 为 备份 数据 库 文件 所 存储 的 位 置 ) 
(20) 重 置 数据 库 密码 ， 修 改 root 用 户 的 密码 。 


| mysqladmin -uroot -p 旧 密 码 password 新 密码 


| 方式 进行 mysq] 的 启动 、 停 止 和 重启 
| service Imysqld start 
service mysqld stop 
service mysqld restart 
4. Linux 下 发 布 网 站 
| Linux 下 发 布 基于 PHP 的 网 站 , 首先 需要 配置 PHP 的 运行 环境 , 其 次 需要 对 网 卡 参数 进行 
| 设 定 。 本 实例 将 以 发 布 “ 读 者 之 家 ”网 站 为 例 讲解 Linux 下 如 何 实现 网 站 的 发 布 。 
| (1) 假设 已 经 申请 到 了 以 下 网 络 参数 : 
IP: 192.168.1.* 
Netmask: 255.255.255.* 
Network: 192.168.1.0 
Broadcast: 192.168.1.* 
Gateway: 192.168.1.* 
主机 名 : mingrisoft 
DNS: 168.95.1* 
(2) 将 “读者 之 家 ”网 站 的 所 有 文件 复制 到 Apache 主 目录 下 。 
(3) 设置 主机 名 称 。 在 终端 窗口 中 输入 如 下 命令 编辑 /etc/sysconfig/network 文件 。 
vi /etc/sysconfig/network 


将 该 文件 中 的 参数 NETWORKING 设置 为 yes， 表 示 启 动 网 络 ， 将 参数 HOSTNAME 设置 
mrsoft， 表 示 设 置 主机 名 为 “mrsoft”。 
(4) 设置 网 卡 参数 。 在 终端 窗口 中 输入 如 下 命令 编辑 文件 /etc/sysconfig/network-scripts/ 
ifcfg-eth0。 
vi /etc/sysconfig/network-scripts/ifcfg-ethO 


| 按 如 下 形式 设置 该 文件 的 相关 参数 : 

| DEVICE=eth0 设置 网 卡 名 称 ， 要 与 这 fg-eth0 对 应 。 
| ONBOOT=yes 指定 在 开机 时 启动 网 卡 。 

| BOOTPROTO=static 设 定 启动 时 获取 卫 的 方式 。 
IPADDR=192.168.1.* 设 定 服务 器 了 地 址 。 
NETMASK=255.255.255.* 设 定子 网 掩 码 。 
BROADCAST=192.168.1.* 设 定 同 网 段 的 广播 地 址 。 
GETWAY=192.168.1.* 设 定 网 卡 的 网 关 。 

| (5) 设置 DNS 主机 的 他。 在 终端 编辑 /etc/resolv.conf 文件 。 
| vi /etc/resolv.conf 


注意 : 
”2% | Linux 下 启动 和 停止 MySQL 服务 器 的 命令 是 (使 用 rpm 包 安 装 的 mysql, 可 以 使 用 下 面 的 
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设置 参数 nameserver 的 值 为 168.95.1.*。 
(6) 重新 启动 网 络 设置 。 在 终端 窗口 中 输入 如 下 命令 : 


/etc/re.d/inin.d/network restart | 
ifdow eth0 | # 
ifiup eth0 | 策 - 


(7) 打开 浏览 器 ， 在 地 址 栏 中 输入 服务 器 卫 地 址 或 域名 ， 如 果 出 现 如 图 1.18 所 示 页 面 ， 
则 说 明 Linux 下 “读者 之 家 ”网 站 发 布 成 功 。 | 
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图 1.18 在 Linux 下 发 布 网 站 


专家 点 评 
在 Linux 下 配置 PHP 开发 环境 需要 注意 以 下 几 个 问题 : | 
(1) 首先 需要 打开 Linux 终端 (Linux 下 几乎 所 有 的 软件 都 需要 在 终端 下 安装 )。 在 | 
RedHat 9 中 选择 “ 主 菜单 ”一 “系统 工具 ”一 “终端 ”命令 。 | 
(2) 在 安装 PHP5 之 前 ， 需 要 首先 查看 libxml 的 版 本 号 。 如 果 libxml 版 本 号 小 于 2.5.10，， 
则 需要 先 安装 libxml 高 版 本 。 | 
(3) 在 Linux 下 配置 Apache 和 PHP， 同 样 需要 配置 php.ini 和 httpd.conf。 


问题 3 ”如何 搭建 [IS+PHP+MySQL 环境 ? 


问题 讲述 

Intermet 信息 服务 器 (Intemet Information Server， 缩 写 为 IS) 是 Microsoft 的 Web 服务 器 。 
它 集 成 于 Windows NT Server 之 中 ， 方 便 易 用 ， 为 Web 应 用 程序 提供 了 功能 强大 的 运行 平台 。 | 
那么 该 如 何 应 用 IIS 平台 来 搭建 一 个 PHP 的 开发 环境 呢 ? 
专家 解答 

1. 安装 IIS 

JIS 目前 的 最 高 版 本 是 Windows Vista 中 安装 的 IIS 7.0， 而 Windows 2003 中 的 是 IIS6.0， | 
Windows XP 等 使 用 的 则 是 更 低 的 版 本 。 本 章 以 Windows 2003 的 IIS 6.0 为 例 来 讲解 IS 的 安装 ，， 
其 他 版 本 的 安装 和 使 用 基本 相同 。 | 

(1) 选择 “开始 ”一 “设置 ”一 “控制 面板 ”命令 ， 在 弹出 的 窗口 中 选择 “添加 /删除 程 | 
Se L_ 


人 ip 六 预 知道 的 300 个 问题 


| 序 ” 窗口 。 单 击 左 侧 的 “添加 /删除 Windows 组 件 ”按钮 ， 将 会 弹出 “Windows 组 件 向 导 ” 窗 
| 口 。 在 该 窗口 中 选中 “应 用 程序 服务 器 ” 复 选 框 ， 如 图 1.19 所 示 。 


(2) 进入 “应 用 程序 服务 器 ”窗口 ， 选 中 “Intemet 信息 服务 (IS)” 复 选 框 ， 如 图 1.20 


| 所 示 。 
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图 1.19 选中 应 用 程序 服务 器 图 1.20 选中 Intemet 信息 服务 器 (ITS) 


(3) 进入 “Internet 信息 服务 (ITS)” 窗 口 ， 选 中 “万 维 网 服务 ” 复 选 框 如 图 1.21 所 示 。 

(4) 进入 “万 维 网 服务 ”窗口 ， 选 中 “万 维 网 服务 ” 复 选 框 ， 如 图 1.22 所 示 ， 然 后 单 击 
“确定 ”按钮 ， 开 始 组 件 安装 。 需 要 注意 的 是 ， 在 安装 过 程 中 需要 将 系统 盘 放 到 光驱 中 。 

(5) 安装 完毕 后 ， 就 可 以 使 用 HS 了 。 如 果 之 前 没有 安装 Apache 服务 器 或 Apache 服务 


器 的 端口 已 经 被 修改 了 ， 那 么 只 要 打开 正 浏览 器 ， 并 在 地 址 栏 中 输入 “http://localhost” 就 可 
| 以 看 到 IIS 运行 页 面 。 
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图 1.21 选中 万 维 网 服务 图 1.22 选中 万 维 网 服务 


2. 配置 IS 


下 面 对 IS 进行 配置 ， 具 体操 作 如 下 : 
(1) 选择 “开始 ”一 “设置 ”一 “控制 面板 ”命令 ， 在 弹出 的 窗口 中 双击 “管理 工具 ” 


| 文件 夹 图 标 。 在 弹出 的 窗口 中 双击 刚刚 添加 的 “Intemet 信息 服务 (ITS) 管 理 器 ”图 标 ， 将 弹出 
| IIS 管理 器 窗口 ， 如 图 1.23 所 示 。 


(2) 图 1.23 中 的 默认 网 站 就 是 IIS 服务 器 的 文件 目录 了 。 右 击 “ 默 认 网 站 ”节点 ， 在 弹 


出 的 快捷 菜单 中 选择 “属性 ”命令 ， 对 网 站 的 属性 进行 设置 ， 如 图 1.24 所 示 。 


(3) 最 先 看 到 的 选项 卡 是 “网 站 ”选项 卡 ， 这 里 主要 设置 TCP 端口 号 ， 将 TCP 端口 号 设 


| 置 为 “8080”。 设置 完成 后 单 击 “ 主 目录 ”标签 ， 进 入 如 图 1.25 所 示 的 “ 主 目录 ”选项 卡 。 
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图 1.24 网 站 属性 1.25 主 目录 选 项 卡 


(4)“ 主 目录 ”选项 卡 中 需要 设置 的 地 方 主要 有 3 处 。 第 1 处 是 本 地 路 径 ， 用 户 可 以 自 
行 定义 。 如 F'\AppServwwwmr; 第 2 处 是 设置 执行 权限 ， 设 置 为 纯 脚 本 ; 第 3 处 是 添加 PHP 
的 支持 。 单 击 该 选项 卡 中 的 “配置 ”按钮 ， 将 弹出 “应 用 程序 配置 ”窗口 ， 如 图 1.26 所 示 。 

(5) 可 以 看 到 ， 在 映射 选项 卡 中 可 以 添加 、 编 辑 应 用 程序 扩展 。 单 击 “ 添 加 ”按钮 ， 将 
添加 如 图 1.27 所 示 的 信息 。 设 置 完 毕 后 ， 单 击 “确定 ”按钮 返回 。 


图 1.26 应 用 程序 配置 图 1.27 添加 PHP 扩展 


| 

(6) 打开 “文档 ”选项 卡 ， 添 加 一 个 php 的 默认 页 ， 如 图 1.28 所 示 。 
(7) 关闭 “默认 网 站 属性 ”面板 ， 在 TIS 管理 器 中 选择 “Web 服务 扩展 ”选项 。 在 右 侧 | 
窗口 中 选择 “所 有 未 知 ISAPI 扩展 ”服务 ， 单 击 左 侧 的 “允许 ”按钮 ， 启 动 该 服务 。 设 置 后 的 | 
窗口 如 图 1.29 所 示 。 | 


全 UD 


到 人 ip 六 预 知道 的 300 个 问题 
(8) 关闭 所 有 窗口 ， 重 启 IS 服务 器 。 


默 以 网 站 屋 考 EE 


CE [ll 


3 il 元 洋 
Taternet 数据 于 按 器 。 亿 活 
后 | @ 单 击 允 许 按钮 | 四 mesxm 有 


EDS reb 且 3 扩 民 @ 选 中 该 服务 


一 a 


DS 

CE Ye CS 

9 

me | wh | Aw | wh | = l 
图 1.28 添加 默认 页 图 1.29 开启 ISAPI 扩 展 
3. 配置 PHP 
(1) 将 PHP5 的 安装 文件 php-5.1.1-Win32.zip 解压 到 相应 目录 (如 C:php\、F:Wphp\ 等 )， 
这 里 将 其 放置 在 Fi\php\ 目 录 下 。 


(2) 将 Fi\php\ 目 录 下 的 ibMySQL.dll 文件 复制 到 C:\Windows\system32\ (如 果 是 Windows 
2000， 则 为 CNWINNTsystem32\) 目录 下 。 

(3) 将 Fi\php\ 目 录 下 的 php.ini-dist 文件 复制 到 C:\Windows\ (如果 是 Windows 2000， 则 
为 C\WINNT\) 目录 下 。 然 后 将 php.ini-dist 重 命名 为 php.ini。 

(4) 打开 php.ini 文件 并 定位 到 extension dir ="./" 这 一 行 ， 并 修改 为 extension_dir=" 
F:\php\ext "。 

(5) 定位 到 “;extension=php_MySQL.dll” 这 一 行 ， 将 前 面 的 分 号 “;” 去 掉 。 这 样 ，PHP 
就 可 以 支持 MySQL 数据 库 了 。 如 果 想 要 加 载 其 他 动态 库 ， 其 方法 与 此 相同 。 

(6) 保存 配置 。 

创建 phpinfo.php 文件 ,将 其 保存 在 IS 主 目录 中 指定 的 文件 夹 下 , 测试 PHP 与 IS 是 否 配 
置 成 功 。 代 码 如 下 : 

<?php phpinfo0; ?> 

(7) 在 浏览 器 中 输入 “http://localhost:8080/phpinfo.php”， 如 果 正 确 ， 则 显示 如 图 1.30 所 

示 的 页 面 。 


四 
Ose :DO- 国 国信 当 检 全 :DD 
EQ [be /ocdest 9000/ neinte ate DW Wa” 


怕 E + 由 et 加 
图 1.30 测试 成 功 
。16 。 
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4. 安装 MySQL 

MySQL 是 一 款 备 受用 户 欢迎 的 数据 库 ， 由 于 其 具有 开源 的 特性 ， 所 以 市 场 占有 率 较 高 ， 
得 到 PHP 开发 者 的 青睐 ， 一 直 被 认为 是 PHP 的 最 佳 搭档 。MySQL 数据 库 的 安装 步骤 如 下 : 
(1) 双击 运行 下 载 后 的 程序 (mysql-5.5.12-win32.msi)， 如 图 1.31 所 示 。 

(2) 单 击 “ 运 行 ”按钮 ， 弹 出 如 图 1.32 所 示 的 对 话 框 。 


EE 
| 
| 
! 
| 
| 
| 
| 
1 
1 
| 
1 
| 
| 
| 
| 
| 
| 


Welcome to the MySQL Server 5.5 setup 


EO 


无 法 验证 发 布 者 ”您 确定 要 运行 此 软件 吗 ? 


Te set rs seve $5 for, 
mpeier Ok Nexttcaninue 9 Concel 加 etthe Secup 
ad 


包 称 :Dimysql-5.513-wir32.msi 
发 行商 。 未 知 发 布 者 


类 型 。 Wirdows Installer 三 序 包 
发 送 方 Dewnysql-5.5.13-wir32.msi 


(| > 
回 打 开 此 文件 前 总 是 调 问 of MUSQL: 全 可 和 ai ome rader to ato 和 
【庆生 ET 
图 1.31 询问 是 否 打开 文件 图 1.32 开始 运行 MySQL 安装 向 导 


(3) 单 击 Next 按钮 ， 弹 出 如 图 1.33 所 示 的 对 话 框 。 
(4) 单 击 Next 按钮 ， 弹 出 如 图 1.34 所 示 的 对 话 框 。 
SS 一 一 | Wav 


‘End-User License Agreement 
Pease read the folowing loenoe agreement carefhy 


acceps me werms in the tcance Agreement 


Ca) Coe Ce ] (Cems) 


图 1.33 询问 是 否 接受 协议 图 1.34 选择 安装 类 型 


(5) 单 击 Typical 按钮 ， 弹 出 如 图 1.35 所 示 的 对 话 框 。 
(6) 单 击 Install 按钮 开始 安装 ， 安 装 进度 如 图 1.36 所 示 。 


其 MSOLRve 55 5etup 划 WiQLsvesssup | 
aa to me moot severs,s I | wotsira moot seners's 
bao te rialetny Cheh Be borrow er here wry fro | a nh es ar 
ods Ca ene 


Sate opyngmen je 
一 一 ] 


图 1.35 确认 前 面 各 选择 步骤 的 对 话 框 1.36 ”MySQL 安装 进度 对 话 框 
。17 。 
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(7) 在 安装 完成 后 会 弹出 如 图 1.37 和 图 1.38 所 示 的 两 个 广告 对 话 框 ， 在 广告 对 话 框 〈 一 ) 
中 单 击 Next 按钮 。 


MySQL Enterprise 


We AMysak Enterprse ubrer phon Ethe most 
MuUSQL.; ns pane Rs 
Ene" 


1.Thel 


图 1.37 广告 对 话 框 〈 一 ) 图 1.38 广告 对 话 框 〈 二 ) 
(8) 在 广告 对 话 框 〈 二 ) 中 单 击 Next 按钮 ， 弹 出 如 图 1.39 所 示 的 对 话 框 。 
(9) 选中 Launch the MySQL Instance Configuration Wizard 复 选 枉 ， 单 击 Finish 按钮 ， 弹 
出 如 图 1.40 所 示 的 对 话 框 。 
项 MsqLsevesssup | 


Completed the MYSQL Server 5.5 Setup 
Wizard 


MysSQL Seer Instarce Configurtiorwiard 


to Ure HySQL server tustance 
oat Wa 


Chic he Fnenbucon to ot he Seap Waard, 


Fiore the WO Yrotoree Coriarsson Weord ee 


= ER Co Concel 
图 1.39 ”安装 完成 对 话 框 图 1.40 开始 对 MySQL 数据 库 进 行 配置 


(10) 单 击 Next 按钮 ， 显 示 如 图 1.41 所 示 的 对 话 框 。 

(11) 选中 Detailed Configuration 〈 详 细 配置 ) 单 选 按钮 ， 单 击 Next 按钮 ， 显 示 如 图 1.42 
所 示 的 对 话 框 。 
TREE 


MysQL Server instance Confiouration 
Conmguretne MsQL Srver saserver nstance, 


人 Detailed 


Choose thk coniiguration bFeto araate the optimal yerer 


Mn 5a erelopment macmne, tn mory cener appncatiors wt 
Derun on 
setup for this machine、 y 


MI Saner smbud on Wsea ninlal nount or 


Several senerappliatinns will he rinnng anthis machine 
hoce this npfinr fnr weh/anpliration cemere. MSGI willhave 
standard Configuration meditm memory usage. 
Use this orly onmacrines :hat do noraready have a My5QL 
Semer installatian. This will use 2 general purpore configuration 让 Dedicated mysQl Server Marhine 
“orthe sererthat car be tuned mancaly. = 


ea ma Back Gn 
1.41 选择 使 用 哪 种 配置 方式 图 1.42 选择 服务 器 类 型 
。18 。 
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(12) 选中 Developer Machine (开发 者 机 器 〉 单 选 按钮 ， 单 击 Next 按钮 ， 打 开 如 图 1.43 | 
所 示 的 对 话 框 。 | 
(13) 选中 Multifunction Database( 多 功能 数据 库 ) 单 选 按钮 ， 单 击 Next 按钮 ， 打 开 如 | 
图 1.44 所 示 的 对 话 框 。 | 
MySQL Server Instance Configuration Wizard = 


MySQL Server nstance Configuration 
Configure the MySQL Server 5.5 sever in stance, 


MySQL Server Instance Configuration Wizard 


MySQL Server Instance Configuratior 
Configure the MWSQL Sever 5.5 server instance. 


Please select he database usage. Please select the drive for the InnoDB datafile, ff you do not want to usethe default 


GT Bnd Tabepwe scttnwv 
General purpose databases, This will optimize the serverforthe 所 Pease choose the drive and directory where the InnoDB 
use of the fast tansactional InnoD storage engine and the tablespace should be placed. 


high speed MyISAM storage engine 


Transactional Database only Ie 可 fnstanaton Path | 
Optimized for application severs and tansadional web 
| sppleatons THF wl make innoDe the wa storage engine. 
Note that the MSAM engine Gan sal be usedl. Nabe td 
File System: NTFS 


三 Won-Transactional Database Only 
Suted for simple web applications monitoring or logging 
本 appteatons 3 wellas ralysts programs. OnY the 


‘Drve nro 
non-transactional MISAM storage sngine will be activated. 


国 :8s ce Dsspe Used 。 口 111 68 Free Dekapace 


Ce ene | 


人 二 二 


图 1.43 选择 数据 库 类 型 图 1.44 选择 InnoDB 表 空 间 保存 位 置 


(14) 使 用 默认 设置 ， 单 击 Next 按钮 ， 打 开 如 图 1.45 所 示 的 对 话 框 。 
(15) 使 用 默认 设置 ， 单 击 Next 按钮 ， 打 开 如 图 1.46 所 示 的 对 话 框 。 | 


MW corver Eat Carle [> MySQL Server Instance Configuration Wizard 一 一 ! 
| 

MySQL server nstance Configuraton MysQt Server Iinstance Configuration ! 
Configure the MySQL Sever 5.5 semver instance, Configure the MWSQL Sever 5.5 serrer instance, ! 

| 

Please set tht approximate number of concurrent connections to the server. Please set the naworking options, | 

& Desion support (OSs)/OLAR (7 Enable TCP/IP Networking | 
Seled this option for database applcations that will not require Enable this to allowTCPHP connections, When disabled only ! 

总 Pe hmmnen teedton A mnier Wf 20 b tocal connections through named pipes are allowed, | 
tonnections wil be assumed. | 


Online Transaction Processing (OLTP) Pat Number 末了 厂 Addfirewall exception torthis port 
hoose this opton for highly concurent apphcations that nay 
have at any one hme up to 500 acthe connections such as 


heavily loaded web servers, Please set the sever SQL mode. 


| Enable Strict hodc 
总 Please enter the approxmate number of concurent a 
| obese somer hls tecommen ded to noble ts odon 


ack Caneel | -| mr | canal 


图 1.45 选择 服务 器 并 发 访问 人 数 图 1.46 设置 端口 号 和 服务 器 SQL 模式 


注意 : 
MySQL 使 用 的 默认 端口 是 3306， 在 安装 时 也 可 以 修改 为 其 他 的 端口 号 ， 如 3307。 但 是 一 
般 情 况 下 ， 不 要 修改 默认 的 端口 号 ， 除 非 3306 端口 已 经 被 占用 。 


(16) 使 用 默认 设置 ， 单 击 Next 按钮 ， 打 开 如 图 1.47 所 示 的 对 话 框 。 
(17) 选中 Manual Selected Default Character Set/Collation 单 选 按钮 ， 设 置 字符 集 编码 为 
utf8， 单 击 Next 按钮 ， 进 入 如 图 1.48 所 示 的 对 话 框 。 
(18) 选中 Install As Windows Service 和 Include Bin Directory in Windows PATH 复 选 框 ， 
生 Next 按钮 ， 进 入 如 图 1.49 所 示 的 对 话 框 。 
(19) 输入 数据 库 的 密码 “111”， 单 击 Next 按钮 ， 打 开 如 图 1.50 所 示 的 对 话 框 。 
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图 1.47 设置 默认 的 字符 集 图 1.48 针对 Windows 系统 进行 的 设置 


MySQL Server Instonce Configuretion Waord 到 | 


ysQL Server Instance connguraoon 
Contigure the MySQ. Server $5 sever natance 


MySQL Server nstence Configuretion Wizard 


MySQL Server Instance Connguraoon 
Confgurn the MSQ Sever 5.5 saver nctance 
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图 1.49 输入 数据 库 的 密码 图 1.50 确认 配置 对 话 框 


注意 : 
在 安装 MySQL 数据 库 时 ， 一 定 要 牢记 在 上 述 步骤 中 设置 的 默认 用 户 root 的 密码 ， 这 是 在 
访问 MySQL 数据 库 时 必须 使 用 的 。 


(20) 单 击 Execute 按钮 ， 执 行 前 面 进行 的 各 项 配置 ， 过 程 如 图 1.51 所 示 。 配 置 完成 后 的 
界面 如 图 1.52 所 示 。 


MSQ Garver Inatance Confouration Wicard ED MS Server Inctonce Configuration Wicord 


BD mepare contiguntion 
TD we onfiguraton fie fipogam FewaatwrsaLseve Snr 
DO Startsenice 

D Appy security sedings 


图 1.51 显示 配置 进度 图 1.52 完成 配置 


至 此 ，MySQL 安装 成 功 。 如 果 要 查看 MySQL 的 安装 配置 信息 ， 则 可 以 通过 MySQL 安装 
目录 下 的 myini 文件 来 完成 。 
在 myini 文件 中 可 以 查看 到 MySQL 服务 器 的 端口 号 、MySQL 在 本 机 的 安装 位 置 `.MySQL 
数据 库 文件 存储 的 位 置 以 及 MySQL 数据 库 的 编码 等 配置 信息 。 参 考 内 容 如 图 1.53 所 示 。 
.20 。 
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图 1.53 myini 文件 的 配置 信息 


专家 点 评 


(1) 如 果 在 机 器 中 同时 安装 了 IS 和 Apache， 则 应 对 IIS 或 Apache 的 端口 号 进行 修改 ， 
以 避免 这 两 个 服务 器 发 生 冲 突 。 | 
(2) 如 果 本 机 使 用 IS 服务 器 配置 PHP 的 开发 环境 ， 那 么 就 不 要 再 使 用 集成 化 的 安装 包 | 
来 配置 。 因 为 它们 会 发 生 冲 突 ， 从 而 导致 配置 不 能 生效 。 | 
(3) 在 安装 PHP 的 过 程 中 应 注意 ， 一 定 要 将 PHP 文件 夹 下 的 libMySQL.dll 复制 到 系统 根 | 
目录 下 的 Windows/system32 文件 夹 下 。 | 
(4) IIS 服务 器 的 伪 静 态 设置 (UrlRewrite 地 址 变换 功能 )。 | 
通过 地 址 变换 可 以 实现 页 面 静态 功能 , 方便 搜索 引擎 收录 , 例如 , http://***/news.php?id=11 | 
可 以 变 为 htt//***/news-11.html， 同 时 还 隐藏 了 开发 标记 ， 避 免 每 个 页 面 都 使 用 .php 等 后 级 ， | 
提高 了 网 站 的 安全 性 。 
注意 ，IIS 默认 是 不 支持 这 个 功能 的 ， 需 要 单独 安装 插件 。 其 具体 操作 步骤 如 下 : 
中 进入 http://www.helicontech.com/download-isapi rewrite3.htm 网 站 。 
@ 下 载 ISAPI Rewrite 3 Lite installation package( 精 简 版 )。 
@ 安 装 ISAPI Rewrite3_0048_Lite.msi。 | 
@ 打 开 IS, 在 Web 属性 的 “ISAPI 筛选 器 ”中 添加 位 于 安装 目录 的 HeliconSAPI Rewrite3 | 
文件 夹 中 的 ISAPI Rewrite .dll， 名 称 自行 设置 。 | 
至 此 ，IS 的 插件 安装 配置 成 功 。 运行 位 于 安装 目录 的 HeliconNSAPI Rewrite3 文件 夹 中 的 | 
Helicon Managerexe， 编 辑 正则 表达 式 ， 实 现 对 URL 地 址 的 转换 。 同 时 将 文件 保存 在 实例 根 目 | 
录 下 ， 并 命名 为 “.htaccess”。 | 


问题 4 PHP 集成 开发 环境 的 特点 有 哪些 ? 


问题 讲述 


PHP 集成 开发 环境 软件 有 很 多 ， 如 XAMPP、EasyPHP 和 Apperv 等 ,那么 每 个 集成 开发 环 


专家 解答 
1. XAMPP | 
XAMPP 是 一 个 功能 全 面 的 AMPP (Apache、MySQL、PHP、Perl) 软件 包 ， 这 是 Linux | 


*。21 。 


到 人 ip 六 预 知道 的 300 个 问题 


平台 上 可 以 使 用 的 几 种 非 商 业 XAMPP 中 间 件 之 一 。 采 用 这 种 紧密 的 集成 ，XAMPP 可 以 运行 
任何 程序 ， 从 个 人 主页 到 功能 全 面 的 产品 站 点 〈 注 意 ， 这 仅仅 用 于 开发 目的 ， 出 于 安全 考虑 ， 
XAMPP 还 不 适 于 用 在 产品 服务 器 上 )。 

2. EasyPHP 

EasyPHP 是 一 个 Windows 下 的 Apache+Mysql+Perl/PHP/Python 开发 包 , 包 中 集成 了 PHP、 
Apache 和 MySQL， 同 时 也 集成 了 一 些 辅助 的 开发 工具 ， 如 数据 库 管 理工 具 ，phpMyAdmin 和 
php 调试 工具 Xdebug， 无 须 配 置 即 可 运行 。EasyPHP 是 由 法 国人 开发 ， 经 过 EasyPHP 整合 后 


| 的 Apache、MySQL 及 PHP 精简 很 多 ， 运 行 速度 比 独立 安装 的 Apache、MySQL 及 PHP 相对 
| 较 快 且 较 为 稳定 。 目前 最 新 版 本 是 EasyPHP 5.3.6.0、PHP 5.3.6 VC9、 Apache 2.2.17 VC9、MySQL 
| 5.5.10、Xdebug。 


3. AppServ 
如 果 用 户 的 本 地 机 器 没有 安装 过 PHP、MySQL 等 系统 ， 那 么 使 用 该 软件 则 可 以 帮助 用 户 


| 迅速 措 建 完整 的 底层 环境 。 
| 专家 点 评 


基于 上 述 3 个 集成 的 开发 环境 安装 软件 ， 笔 者 认为 还 是 使 用 AppServ 要 好 一 些 ， 因 为 使 用 


| XAMPP 和 EasyPHP， 在 应 用 jpgraph 开发 图 像 时 会 出 现 一 些 错误 。 而 使 用 AppServ 就 不 会 有 
| 任何 的 问题 。 建 议 版 本 为 AppServ 5.10。 


对 于 PHP 开发 环境 的 搭建 ， 最 好 的 方法 还 是 个 人 措 建 ， 搭 建 的 操作 步骤 在 网 上 完全 可 以 


| 找到 ， 只 是 在 搭建 成 功 后 需要 对 其 进行 一 些 修改 ， 并 开启 一 些 PHPini 的 设置 。 例 如 : 


display_error=ON 

输出 错误 信息 。 但 是 如 果 是 在 互联 网 的 服务 器 中 使 用 ， 那 么 就 要 关闭 该 属性 。 
log_errors=on 

在 日 志文 件 中 输出 错误 信息 。 
error_reporting = E_ALL & ~E NOTICE 

设置 错误 级 别 。 


问题 5 如 何 应 用 AppServ 搭建 PHP 开发 环境 ? 


问题 阐述 


AppServ 将 Apache、PHP、MYySQL 和 phpMyAdmin 等 服务 器 软件 和 工具 安装 配置 完成 后 


| 打包 处 理 。 开 发 人 员 只 要 到 网 站 上 下 载 该 软件 后 安装 ， 即 可 完成 PHP 开发 环境 的 快速 搭建 。 
| 这 种 搭建 PHP 开发 环境 的 方法 非常 适合 初学 者 ， 那 么 应 用 AppServ 搭建 PHP 开发 环境 的 步骤 
| 有 哪些 呢 ? 


| 专家 解答 


应 用 AppServ 集成 化 安装 包 搭建 PHP 开发 环境 的 操作 步骤 如 下 : 
。22 。 
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(1) 双击 AppServ-win32-2.5.7.exe 文件 ， 打 开 如 图 1.54 所 示 的 AppServ 启动 页 面 。 
(2) 单 击 Next 按钮 ， 打 开 如 图 1.55 所 
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Welcome to the AppServ 2 5.10 
Setup Wizard 
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图 1.54 AppServ 启动 页 面 图 1.55 AppServ 安装 协议 


(3) 单 击 TIAgree 按钮 ， 打开 如 图 1.56 所 示 的 页 面 。 在 该 页 面 中 可 以 设置 AppServ 的 安装 
路 径 〈 默 认 安 装 路 径 一 般 为 ，C:\AppServ)， 安 装 完成 后 ，Apache、MySQL、PHP 都 将 以 子 目 
录 的 形式 存储 到 该 目录 下 。 

(4) 单 击 Next 按钮 将 打开 如 图 1.57 所 示 的 页 面 。 在 该 页 面 中 可 以 选择 要 安装 的 程序 和 组 
件 (默认 为 全 选 状 态 )。 
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图 1.56 ”AppServ 安装 路 径 选 择 图 1.57 AppServ 安装 选项 


(5) 单 击 Next 按钮 ， 打 开 如 图 1.58 所 示 的 页 面 ， 该 页 面 主要 用 于 设置 Apache 的 端口 号 。 

(6) 单 击 Next 按钮 ， 打 开 如 图 1.59 所 示 的 页 面 ， 该 页 面 主要 用 于 对 MySQL 数据 库 的 
root 用 户 的 登录 密码 及 字符 集 进 行 设置 ， 这 里 将 字符 集 设置 为 “GB2312 Simplified Chinese”， 
表示 MySQL 数据 库 的 字符 集 将 采用 简体 中 文 形式 。 


图 1.58 Apache 端口 号 设置 1.59 MySQL 设置 
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人 ip 六 预 知道 的 300 个 问题 


-RD 
六 
(7) 单 击 Install 按钮 开始 安装 ， 如 图 1.60 所 示 。 


(8) 安装 完成 后 可 以 在 开始 菜单 的 AppServ 相关 操作 列表 中 启动 Apache 及 MySQL 服 
务 ， 如 图 1.61 所 示 。 


coMm PUTES 


图 1.60 AppServ 安装 页 面 图 1.61 AppServ 安装 完成 页 面 


(9) 安装 好 AppServ 之 后 ， 整 个 目录 默认 安装 在 C\AppServ 下 。 此 目录 下 包含 4 个 子 目 
录 ， 如 图 1.62 所 示 ， 用 户 可 以 将 所 有 网 页 文件 存放 到 “www” 目 录 下 。 

(10) 打开 正 浏览 器 ， 在 地 址 栏 中 输入 “http://localhost/” 或 “http://127.0.0.1/” 如 果 打 
开 如 图 1.63 所 示 的 页 面 ， 则 说 明 AppServ 安装 成 功 。 
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(11) 编写 一 个 PHP 脚本 ， 代 码 如 下 : 
<2php 
echo "欢迎 进入 PHP 的 世界 ! "; 
?> 


将 该 文件 命名 为 textphp， 保 存 到 AppServ 的 www 文件 夹 下 。 然 后 在 正 浏览 器 中 输出 
http://127.0.0.1/text.php。 运 行 结果 如 图 1.64 所 示 。 
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图 1.64 ”PHP 程序 运行 结果 


注意 : 
在 使 用 Apache 搭建 PHP 开发 环境 时 , 必须 确保 在 系统 中 没有 安装 Apache、 PHP 和 MySQL。 
否则 ， 要 先 将 这 些 软件 拖 载 ， 然 后 才能 应 用 Apache。 
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专家 点 评 


在 安装 过 程 中 有 以 下 几 个 步骤 需要 注意 。 | 
(1) AppServ 安装 路 径 的 选择 。 具体 安装 在 哪个 磁 禹 下 用 户 可 以 自行 选 拌 ,也 可 以 使 用 默 | 
认 的 安装 位 置 。 
(2) 在 填写 服务 器 信息 时 ， 需 要 注意 Server Name 中 填写 的 是 计算 机 名 称 ，Administrator | 
Email Address 中 填写 的 是 用 户 的 邮箱 地 址 ; 最 关键 的 是 Apache HTTP Port， 它 设置 的 是 服务 器 
的 端口 ， 默 认 使 用 80， 读 者 也 可 以 更 改 该 计 口 号 。 


注意 : | 
如 果 用 户 使 用 默认 的 80 端口 ,那么 必须 确定 本 机 上 的 80 端口 未 被 占用 。 如 果 用 户 机 器 上 已 | 
安装 了 IIS, 那么 80 端口 被 IIS 占用 了 ， 在 安装 AppServ 时 就 不 能 再 使 用 80 端口 了 。 建议 用 户 | 
在 安装 AppServ 时 使 用 默认 80 端口 ， 如 果 IIS 占用 了 80 端口 ， 用 户 可 以 将 IIS 端口 修改 为 82 | 
或 者 其 他 。 当然 , 如果 不想 修改 IIS 的 端口 ,也 可 以 直接 将 AppServ 默认 的 80 端口 修改 为 其 他 。 | 


(3) 在 MySQL 数据 库 的 设置 中 ， 应 注意 MySQL 数据 库 的 密码 和 字符 集 的 设置 。 这 里 设 | 
置 的 密码 是 用 户 在 以 后 通过 phpMyAdmin 访问 数据 库 时 需要 使 用 的 , 所 以 用 户 应 牢记 设置 的 这 | 
个 密码 。 同 时 该 密码 也 是 程序 中 连接 数据 库 时 所 用 的 密码 。 至 于 编码 格式 ， 建 议 用 户 使 用 | 
GB2312 Simplified Chinese， 也 就 是 简体 中 文 。 | 


问题 6 如 何 通过 XAMPP 配置 PHP 开发 环境 ? 


问题 阐述 | 
XAMPP 是 一 个 易于 安装 且 包 含 MySQL、PHP 和 Perl 的 Apache 发 行 版 ， 那 么 如 何 通过 安 | 
装 XAMPP 配置 PHP 开发 环境 呢 ? | 
专家 解答 
XAMPP 1.7.3 版 的 具体 安装 步骤 如 下 : | 
(1) 运行 .exe 安装 文件 ， 将 弹出 如 图 1.65 所 示 的 操作 界面 。 在 这 里 设置 XAMPP 的 安装 | 


路 径 。 
(2) 安装 路 径 设置 完成 后 ， 单 击 Install 按钮 ， 执 行 XAMPP 的 安装 ， 如 图 1.66 所 示 。 


图 1.65 设置 XAMPP 的 安装 路 径 图 1.66 安装 XAMPP 
。25 。 


(3) aints 成 功 后， 将 弹出 如 图 1.67 所 示 的 操作 界面 。 在 该 界面 中 对 XAMPP 进 


行 设 置 ， 包 括 是 否 创建 桌面 快捷 方式 及 是 否 在 开机 时 启动 XAMPP 等 。 


图 1.67 设置 XAMPP 


至 此 ，XAMPP 安装 、 配 置 完毕 。 


| 专家 点 评 


下 面 对 XAMPP 中 的 配置 文件 进行 简单 介绍 。 

XAMPP 中 各 种 配置 文件 的 存储 位 置 如 下 。 

Apache 基本 配置 : \xampp\apache\conf\httpd.conf 
Apache SSL: .\xampp\apache\conf\ssl.conf 

Apache Perl ( 仅 限 插件 ) : .\xampp\apache\conf\perl.conf 
PHP: .\xampp\php\php.ini 

MySQL: .xampp\mysql\binimy.ini 

phpMyAdmin: .\xampp\phpMyAdmin\config.inc.php 
FileZilla FTP 服务 器 : .\xampp\FileZillaFTP\FileZilla 


加 回回 网 回 园 加 


问题 7 XAMPP 


Linux 版 PHP 集成 化 安装 包 


问题 阐述 


| 专 


| 安装 XAMPP 的 步骤 如 下 : 


在 Linux 操作 系统 下 通过 XAMPP 配置 PHP 的 开发 环境 的 步骤 有 哪些 ? 


家 解答 


在 Linux 操作 系统 下 ， 使 用 集成 软件 XAMPP 的 Linux 版 来 配置 PHP 开发 环境 。Linux 下 


(1) 在 Linux 操作 系统 下 ， 选 择 “ 主 菜单 ”一 “系统 工具 ”一 “终端 ”命令 。 
(2) 在 命令 模式 下 ， 首 先进 入 系统 的 根 目录 下 。 

(3) 通过 mkdir 命令 在 根 目录 下 创建 一 个 opt 目录 。 

(4) 通过 tar xvfz 命令 将 XAMPP 解压 缩 到 opt 目录 下 。 


在 Linux 操作 系统 下 执行 XAMPP 的 解压 缩 时 ， 必 须 进 入 系统 的 根 目 录 下 ， 并 且 要 在 根 目录 


| 下 创建 一 个 子 目 录 ， 然 后 才 可 以 执行 解压 缩 的 命令 ， 否 则 ， 将 提示 无 法 找到 指定 的 目录 或 文件 。 
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(5) 按 回 车 键 ， 执 行 XAMPP 的 解压 缩 ， 直 到 安装 成 功 。 其 具体 使 用 的 命令 如 图 1.68 所 示 。 
(6) 安装 成 功 后 查看 /opt/lampp 目录 ， 如 图 1.69 所 示 。 | 


文件 四 编导 加 。 查看 [9 。 转 于 加。 书 答 名。 帮助 名 


qvp、Aa @% A 


后 退 ”前 革 ”同上 一 级 全 刷新 主 文件 


| 位置， [FEITT -+ 
选中 了 = 人 20 功 
图 1.68 XAMPP 的 安装 图 1.69 查看 安装 的 内 容 | 
(7) 在 Mozilla 浏览 器 中 输入 “http://127.0.0.1/index.php”， 运 行 结果 如 图 1.70 所 示 。 | 


注意 : 

在 Linux 操作 系统 下 安装 XAMPP， 必 须 在 Linux “ 主 菜单 ”一 “系统 工具 ”下 的 “终端 ” | 
命令 中 通过 指令 来 完成 ， 切 记 不 要 使 用 任何 微软 操作 系统 下 的 工具 来 进行 操作 。Linux 下 钙 载 | 
XAMPP 的 命令 : rm -rf /opt/lampp。 | 
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图 1.70 Linux 版 的 XAMPP 


问题 8 Apache 配置 文件 全 解 


问题 曾 述 

PHP 开发 环境 配置 成 功 后 ， 有 关 Apache 服务 器 的 配置 文件 (httpd.conf) 被 存储 于 | 
Apache2.2\conf\ 目 录 下 ， 那 么 通过 该 配置 文件 都 可 以 进行 哪些 操作 呢 ? | 
专家 解答 | 

httpd conf 存储 于 Apache2.2\conf\ 目 录 下 ， 是 Apache 服务 器 的 配置 文件 ， 在 该 文件 中 可 以 | 
修改 Apache 服务 器 的 端口 号 、 根 目录 , 激活 伪 静 态 功 能 , 以 及 配置 Apache 服务 器 的 虚拟 主机 。 | 
配置 文件 的 关键 内 容 如 下 : | 


T 


Ng a 


| & ~~、 Pp 闪 需 知道 的 300 个 问题 
| # 设 置 Apache 服务 器 安装 文件 的 存储 位 置 
| ServerRoot "E:/wamp/apache2" 
| # 设 置 Apache 服务 器 的 端口 号 
| Listen 80 
会 内 | # 开 启 伪 静 态 功 能 
EL | LoadModule rewrite_ module modules/mod rewrite.so 
| # 配 置 PHP 
LoadModule php5_module "E:/wamp/php/php5apache2 2.dll" 
| # 设 置 端口 号 
| ServerName localhost:80 
# 定 义 Apache 服务 器 的 Web 存储 路 径 
| DocumentRoot "E:/wamp/Wwww" 
| # 
| <Directory /> 
Options FollowSymLinks 
AllowOverride All 
Order deny,allow 
Deny from all 
Satisfy all 
</Directory> 
# 定 义 Apache 服务 器 的 Web 存储 路 径 
<Directory "E:/wamp/www"> 
Options Indexes FollowSymLinks 
Order Deny,Allow 
Deny from all 
Allow from 127.0.0.1 
</Directory> 
# 设 置 伪 静态 
<Directory "E:/wamp/apache2/cgi-bin"> 
AllowOverride All 
Options All 
Order allow,deny 
Allow from all 
</Directory> 


# 
设置 Apache 服务 器 的 虚拟 主机 ， 只 需要 在 此 文件 的 最 后 添加 配置 方法 即 可 。 


专家 点 评 


| 在 配置 文件 中 ， 上 述 罗 列 的 配置 项 都 可 以 进行 修改 ， 进 而 达到 读者 想 要 的 效果 ， 切 记 在 完 
| 成 对 配置 文件 的 修改 后 应 保存 文件 ， 并 且 重新 启动 Apache 服务 器 ， 这 样 修改 才 会 生效 。 


| 问题 9 PHPINI 配置 文件 全 解 


问题 阐述 

| PHP 开发 环境 配置 成 功 后 ， 在 Linux 操作 系统 中 php.ini 存储 于 /opt/lampp/etc/php.ini 文件 
| 夹 下 ， 而 在 Windows 操作 系统 中 php.ini 存储 于 系统 盘 的 Windows 文件 夹 下 ， 如 果 使 用 集成 化 
| 的 安装 包 来 配置 PHP 开发 环境 ， 也 有 将 php.ini 存储 于 PHP 安装 文件 夹 下 的 。 那 么 通过 该 配置 
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文件 都 可 以 进行 哪些 操作 呢 ? 


专家 解答 


phpiini 文件 是 PHP 在 启动 时 自动 读 取 的 配置 文件 ， 它 是 一 个 ASCLL 文本 文件 ,分 为 多 个 | 
部 分 ， 每 一 部 分 包括 相关 的 参数 。 每 一 部 分 的 名 称 位 于 最 前 面 的 方 括号 内 ， 接着 是 名 称 对 数字 ， | 


且 每 一 名 称 都 独占 一 行 。 使 用 规则 PHP 代码 ， 对 参数 名 称 非常 敏感 ， 不 能 包含 空格 ， 但 是 参 | 


数 可 以 是 数字 、 字 符 串 或 布尔 逻辑 数 。 分 号 位 于 每 一 行 的 开始 ， 作 为 指定 标记 ， 这 就 使 选择 使 
用 或 不 使 用 PHP 的 这 些 特性 变 得 很 方便 ， 而 无 须 通过 删除 该 行 来 实现 。 对 某 特性 进行 注释 〈 即 
添加 分 号 )， 则 该 行将 不 会 被 编译 执行 。 每 次 修改 完 php.ini 文件 ， 必 须 重 新 启动 Apache 服务 
器 ， 以 使 新 的 设置 生效 。 

php.ini 是 PHP 的 配置 文件 ， 用 于 加 载 各 种 函数 库 、 设 置 错误 级 别 和 设置 服务 器 的 时 间 等 。 
php.ini 文件 的 基本 配置 如 表 1.1 所 示 。 


参 数 


error_reporting 


Tegister globals 


include_path 


extension_ dir 
extension 
file uploads 


upload tmp_dir 


upload max filesize 


表 1.1 php.ini 文 件 的 基本 配置 
说 明 

设置 错误 处 理 的 级 别 。 推 荐 值 为 E ALL & -EE 
NOTICE & ~E_STRICT, 显示 所 有 错误 信息 , 除了 Es ONE SE 
提醒 和 编码 标准 化 警告 
通常 情况 下 可 以 将 此 变量 设置 为 Off, 这 样 可 以 对 通 
过 表单 进行 的 脚本 攻击 提供 更 为 安全 的 防范 措施 
设置 PHP 的 搜索 路 径 ， 该 参数 可 以 接收 系列 的 目 | ww。 aas 
录 。 当 PHP 遇 到 没有 路 径 的 文件 提示 时 ， 它 将 会 自 PP 
动 检测 这 些 目录 ， 需 要 注意 的 是 ， 当 某 些 选项 允许 |nelude Path "php/includes" 
多 个 值 时 ， 应 使 用 系统 列表 分 隔 符 ， 在 Windows 下 | ineows "pathlsWpath2" 
使 用 分 号 “; ”， 在 Linux 下 使 用 冒号 “: ” ;include_path = ".;c:\php\includes" 
指定 PHP 的 动态 链接 扩展 库 的 目录 \ext 目 录 下 
指定 PHP 启 动 时 所 加 载 的 动态 链接 扩展 库 。 PHP 的 |PHP 的 常用 扩展 库 在 初次 安装 配 
常用 扩展 库 及 其 说 明 可 参见 表 1.2 置 后 均 被 注释 ， 需 读者 手动 更 改 


默 认 值 


register_globals = On 


设置 是 否 允 许 通 过 HTTP 上 传 文件 file_uploads=On 
设置 通过 HTTP 上 传 文件 时 的 临时 目录 , 如 果 为 空 ， 本 
则 使 用 系统 的 临时 目录 np 
设置 允许 上 传 文件 的 大 小 ， 如 “50MB”， 必 须 十 


es ~ lupload_max filesize=2MB 
写 单位 i 


post max size 


控制 在 采用 POSTO 方 法 进行 一 次 表单 提交 中 PHP 
所 能 够 接收 的 最 大 容量 。 要 上 传 更 大 的 文件 ， 则 该 
值 必须 大 于 upload_max_filesize 的 值 。 

如 upload max filesize=10MB ， 那 么 upload max 
filesize 的 值 必须 要 大 于 10MB 


post_ max_size = 8MB 


max_input_time 


以 秒 为 单位 对 通过 POST、GET 以 及 PUT 方 式 接收 


数据 时 间 进 行 限制 站 
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表 1.2 PHP 常用 扩展 库 及 其 说 明 
扩 展 库 说 明 
php ftp.dll 支持 FTP 函 数 库 ， 可 以 实现 客户 机 与 服务 器 之 问 标准 传送 协议 (FTP) 
php gd2.dll 支持 图 像 处 理 函 数 库 ， 支 持 .gif、.jpg、.png 等 多 种 图 像 格 式 
php imap.dll 支持 imap 电 子 邮件 处 理 函数 库 
php_mssql.dll 支持 MsSQL 数 据 库 
php_msql.dll 支持 mSQL 数 据 库 
php_mysql.dll 支持 MySQL 数 据 库 
php_oracle.dll 支持 Oracle 数 据 库 
php_pdf dl 支持 PDF 文件 处 理 函数 库 
php_sockets.dll 支持 Sockets 处 理 函数 库 
jp zliball 支持 zlib 文 件 压缩 函数 库 
php pdo.dll 支持 PDO 数 据 库 抽 象 层 
php pdo_mysql.dll 支持 MySQL 数 据 库 
php pdo_mssql.dll 支持 MS SQL Server 数 据 库 
php_pdo oci8.dll 支持 Oracle 数 据 库 
php_pdo odbc.dll 支持 ODBC 数 据 库 
php_pdo pgsql.dll 支持 PGSQL 数 据 库 


问题 10 ” Apache 不 能 处 理 PHP 文件 


问题 阐述 


使 用 RPM 安装 了 PHP， 但 Apache 却 不 能 处 理 PHP 文件 ， 为 什么 ? 


专家 解答 


假设 用 户 使 用 RPM 正确 安装 了 PHP， 则 必须 在 httpd.conf 文件 中 添加 如 下 儿 行 代码 : 


#Extra Modules 


AddModule mod php.c 
AddModule mod php3.c 
AddModule mod perl.c 


#Extra Modules 


LoadModule php_module modules/mod php.so 
LoadModule php3_module modules/libphp3.so 
LoadModule perl module modules/libperl.so 


另外 ， 再 添加 : 


AddType application/x-httpd-php3.php3 
到 httpd.conf 的 全 局 部 分 ， 获 得 需要 提供 PHP 支持 的 部 分 。 
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问题 11 无 法 找到 httpd.h 文件 


问题 阐述 | 

PHP 和 Apache 一 起 编译 时 ， 显 示 无 法 找到 httpd.h 文件 ， 但 事实 上 该 文件 是 存在 的 ， 这 个 
问题 怎么 解决 呢 ? | 
专家 解答 


用 户 需 要 让 PHP 配 置 即 安装 脚本 知道 自己 的 Apache 源码 的 最 高 级 目录 ,而 不 是 包含 htpdh | 
文件 的 目录 。 也 就 是 说 , 应 该 指定 "--with-apache=/path/to/apache/" 而 不 是 "--with-apache=/path/to/ | 
apache/src"。 | 


问题 12 Apache 不 能 启动 的 解决 方法 


问题 阐述 | 
由 于 安装 比较 烦琐 ， 有 时 出 错 在 所 难免 ， 以 下 是 几 个 在 安装 中 出 现 的 典型 错误 ， 仅 供 读者 
参考 。 | 
专家 解答 
(1) 在 安装 Web 服务 器 时 提示 “Socket error” 或 “address0.0.0.0: 80 not available”。 
错误 主要 是 由 于 当前 环境 中 已 经 运行 了 其 他 Web 服务 器 (如 IIS),，80 端口 已 eit | 
解决 方案 : 为 IS 分 配 其 他 瑟 地 址 或 停止 , 然后 启动 Apache 的 安装 程序 , 选择 “REPAIR” | 
修复 方式 来 重新 安装 。 | 
(2) 浏览 器 显示 400 的 错误 信息 (或 请 求 )。 这 是 由 于 访问 的 .php 文件 ，PHP 引擎 并 没有 | 
找到 ， 可 能 是 在 httpd.conf 中 设置 的 路 径 不 正确 造成 的 。 | 
LoadModule php5_module "E:/wamp/php/php5apache2 2.dll" | 
解决 方案 : 修改 httpd.con 配置 文件 , 重新 设置 PHP 引擎 路 径 , 保存 后 重启 Apache 服务 器 。 | 


问题 13 ”如 何 解决 Apache 服务 器 端口 冲突 ? 


问题 阐述 
如 何 解决 Apache 服务 器 端口 冲突 ? 
专家 解答 


Apache 服务 器 的 核心 配置 文件 是 httpd.conf， 要 修改 其 端口 号 ， 自 然 也 在 该 文件 中 进行 。 | 
定位 到 Listen 80 一 行 ， 修 改 Listen 的 值 ， 实 现 Apache 服务 器 端口 号 的 更 改 。 切记 修改 后 应 保 | 
存 ， 并 重新 启动 Apache 服务 器 。 | 


3 UD 


3 人 ip 六 预 知道 的 300 个 问题 


| 专家 点 评 


| 芽 服务 器 、 迅雷 的 默认 端口 号 为 80， 与 Apache 服务 器 默认 端口 号 相同 。 由 于 采用 了 相同 
| 的 端口 号 80， 因此， 在 运行 网 页 时 就 会 发 生 冲突 。 


央 一 如 果 用 户 机 器 上 安装 了 IIS 服务 器 ， 就 需要 修改 IIS 的 默认 端口 ， 香 则 将 导致 Apache 服务 
器 无 法 正常 工作 。 更 改 IIS 的 默认 使 听 端口 号 80， 可 以 在 IIS 的 管理 器 中 进行 设置 或 者 停止 IIS 


| 的 服务 。 

| 另外 ， 用 户 也 可 以 在 安装 Apache 服务 器 时 将 默认 的 端口 号 进行 更 改 ， 从 而 解决 两 个 服务 

| 器 或 与 其 他 软件 共用 一 个 端口 号 而 产生 冲突 的 问题 。 如 果 对 Apache 的 端口 号 进行 了 修改 ， 那 

| 么 在 运行 PHP 项 目 时 ， 输 入 的 地 址 就 会 发 生 一 些 变化 。 

| 例如 ， 若 将 Apache 的 端口 号 修改 为 8080， 那 么 在 运行 本 机 的 PHP 项 目 时 ， 应 输入 
“http://127.0.0.1:8080/”。 


问题 14 ”如何 配置 Apache 虚拟 主机 ? 


问题 阐述 


| 在 配置 PHP 运行 环境 时 ， 如 果 是 作为 服务 器 来 使 用 ， 那 么 就 少不了 配置 Apache 的 虚拟 
| 机 。 在 虚拟 主机 的 配置 中 可 以 实现 域名 与 指定 项 目的 对 接 ， 在 网 页 中 输入 指定 的 域名 ， 即 可 访 
| 问 对 应 的 项 目 内 容 。 配 置 Apache 的 虚拟 主机 是 创建 PHP 服务 器 必须 掌握 的 一 项 内 容 ， 那 么 该 
| 如何 配 置 Apache 虚拟 主机 呢 ? 


专家 解答 


| 配置 Apache 虚拟 主机 ， 其 前 提 是 在 作为 服务 器 的 计算 机 中 已 经 正确 地 配置 了 一 个 PHP 运 
| 行 环境 ， 配 置 Apache 虚拟 主机 的 操作 在 Apache 的 httpd.conf 文件 中 进行 ， 其 具体 步 又 如 下 : 
(1) 定位 到 Apache2\conf\ httpd.conf 文件 ， 在 此 文件 中 添加 如 下 内 容 : 
<VirtualHost *:80> 
ServerAdmin any(@any.com 
DocumentRoot "E:\Wwamp\www\mingribook" 
ServerName www.mingribook.com 
ErrorLog "logs/phpchinal.com-errorlog" 
CustomLog "logs/phpchinal.com-access.log" common 
</VirtualHost> 


第 1 行 ， 定 义 虚 拟 服务 器 的 标签 ， 指 定 端口 号 。 

第 2 行 ， 指 定 一 个 邮箱 地 址 ， 可 以 随意 指定 。 

第 3 行 ， 定 义 要 访问 的 项 目 在 Apache 服务 器 中 的 具体 路 径 。 

第 4 行 ， 指 定 服务 器 的 访问 名 称 ， 即 与 项 目 绑 定 的 域名 。 

第 5、6 行 ， 定义 Apache 中 日 志文 件 的 存储 位 置 。 

第 7 行 ， 定 义 虚 拟 服务 器 的 结束 标签 。 
| 上 述 7 行 代 码 即 完成 一 个 虚拟 服务 器 的 配置 操作 ， 如 果 存 在 多 个 域名 ， 并 且 需 要 绑 定 
| Apache 服务 器 下 的 多 个 项 目 , 那么 就 以 此 类 推 , 重复 上 述 操作 ,为 每 个 域名 绑 定 不 同 的 项 目 文 
| 件 即 可 ， 即 修改 DocumentRoot 和 ServerName 指定 的 值 。 
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(2) 在 完成 虚拟 主机 的 配置 之 后 ， 需 要 保存 httpd.conf 文件 ， 且 重新 启动 Apache 服务 器 。 | 
专家 点 评 


如 果 在 Apache 服务 器 中 配置 了 虚拟 主机 , 那么 在 Apache 服务 器 中 将 不 能 再 直接 通过 路 径 | f 
的 方式 访问 其 他 项 目 ， 而 必须 使 用 虚拟 主机 中 指定 的 服务 器 名 称 才能 访问 。 例 如 , 定义 | 图 由 
DocumentRoot 访问 文件 为 "E:Vwampvwwwwmingribook"，ServerName 的 名 称 为 127.0.0.1， 那 么 
在 访问 程序 时 可 以 直接 在 浏览 器 中 输入 127.0.0.1， 即 可 访问 此 项 目的 内 容 。 但 是 ,在 | 
E\wamp\www\ 文 件 夹 下 还 存储 了 另外 一 个 项 目的 文件 夹 mrbccd， 如 果 在 没有 设置 虚拟 主机 的 | 
情况 下 ， 可 以 直接 通过 http://127.0.0.1/mrbecd 的 形式 访问 这 个 项 目 ， 此 时 再 以 这 种 形式 进行 访 | 
问 时 却 不 能 正常 输出 。 | 


问题 15 如 何 动态 加 载 扩 展 库 ? 


问题 阐述 
在 PHP 开发 时 ， 如 果 临 时 需要 一 个 新 扩展 库 ， 该 如 何 加 载 呢 ? 
专家 解答 
看 下 面 的 php.ini 配置 文件 。 
enable_dl=on 
首先 要 确认 上 面 的 enable_dl 的 状态 值 为 on， 然后 在 PHP 程序 汇总 就 可 以 使 用 下 列 方法 。 


dl("gnu_gettext.dl1"): //Win32 环境 下 
dl("gnu_gettext.so"); /Win32 环境 下 


使 用 上 面 的 语法 就 可 以 实现 动态 地 按 需 调 入 扩展 库 , 其 调 入 的 目录 路 径 根 据 php.ini 中 设置 | 
的 extension dir 装 入 ， 从 而 节约 资源 ， 提 高 程序 效率 。 | 
如 果 不 确定 该 php 扩展 库 是 否 装 入 ， 可 用 以 下 代码 进行 处 理 。 


if(extension_loaded("")!=true){ 
dl("gnu gettext.so"):; 
} 


表示 如 果 要 调用 的 扩展 未 加 载 内 存 ， 则 立即 装 入 该 扩展 库 。 
问题 16 ”register globals ( 注册 为 爹 局 变量 ) 的 开启 与 关闭 


问题 阐述 
如 何 判 断 register_globals 全 局 变量 是 开启 还 是 关闭 ? 
专家 解答 


register globals 是 php.ini 中 的 一 个 配置 ， 该 配置 直接 影响 PHP 如 何 接收 和 传递 参数 。 
如 果 它 被 设置 为 Off， 那 么 在 获取 通过 POSTO 或 者 GETO 方 法 传递 的 参数 时 ， 就 需要 使 用 | 
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$_POST['user name'] 或 $ GET[user name']。 

如 果 它 被 设置 为 On， 那 么 就 可 以 直接 使 用 $user_ name 来 获取 传递 的 值 。 

它 的 设置 不 只 影响 form、url 参数 的 传递 ， 同 时 也 影响 到 了 SESSION 和 Cookie。 同 样 ， 当 
它 的 值 为 OQ 任 时， 获取 SESSION 和 Cookie 的 值 就 需要 使 用 $_ SESSION[] 和 $_COOKIE[]; 而 当 
它 的 值 为 On 时 ， 就 可 以 直接 通过 变量 名 称 获取 变量 传递 的 值 。 


技巧 : 
当 用 户 遇 到 获取 不 到 form 表单 中 传递 的 值 时 ， 可 以 查看 使 用 的 方法 是 POST 还 是 GET， 


| 并 且 确 认 使 用 的 变量 名 称 是 否 正确 。 另 外 ， 可 以 查看 一 下 php .ini 文件 ， 看 register globals 是 开 
| 启 还 是 关闭 ， 确 定 在 获取 参数 值 使 用 的 方法 有 效 。 


| 专家 点 评 


建议 读者 在 配置 PHP 的 开发 环境 时 ， 将 register_ globals 设置 为 Off， 有 两 个 好 处 : 第 一 可 


| 以 提高 代码 的 安全 性 ; 第 二 可 以 提高 代码 的 可 兼容 性 。 


用 户 现在 使 用 的 所 有 PHP 服务 器 中 ，register_globals 都 是 设置 为 O 任 的 ， 所 以 一 旦 用 户 的 


程序 在 开发 时 设置 的 是 On， 那 么 当 它 上 传 到 服务 器 时 ， 就 会 出 现 问题 。 在 PHP5 中 ,用户 在 配 
| 置 时 还 有 选择 的 余地 ， 可 以 选择 是 On 或 Off,， 但 是 在 PHP6 中 这 项 内 容 已 经 被 删除 ， 所 以 用 户 
| 必须 做 到 未 雨 绸 纽 ， 养 成 一 个 良好 的 编程 习惯 。 


问题 17 为 什么 要 设置 时 区 ? 


| 问题 阐述 


当 安 装 完 PHP 开发 环境 后 ， 获 取 系 统 时 间 时 ， 会 发 现 获 取 到 的 时 间 与 系统 时 间 不 对 应 。 


那么 如 何 对 时 区 进行 设置 ， 并 获取 到 相对 应 的 时 间 呢 ? 
| 专家 解答 


要 获取 本 地 当前 的 时 间 ， 则 必须 更 改 PHP 语言 中 的 时 区 设置 。 更 改 PHP 语言 中 的 时 区 设 


| 置 有 两 种 方法 。 


1. 在 php.ini 文 件 中 设置 时 区 

在 php.ini 文件 中 , 定位 到 [date] 下 的 “:date.timezone =” 
选项 ， 去 掉 前 面 的 分 号 ， 并 设置 其 值 为 当地 所 在 时 区 使 用 
的 时 间 ， 修 改 内 容 如 图 1.71 所 示 。 

例如 ， 如 果 当 地 所 在 时 区 为 东 作 区， 那么 就 可 以 设置 、 
“date.timezone =” 的 值 为 PRC、Asia/Hong Kong、Asia/ 图 171 设置 PHP 的 时 区 
Shanghai (上 海 ) 或 Asia/Urumqi (乌鲁木齐 ) 等 ， 这 些 都 是 东 八 区 的 时 间 。 

设置 完成 后 ， 保 存 文件 ， 重 新 启动 Apache 服务 器 即 可 。 

2. 通过 date_default timezone set 函数 设置 时 区 

在 应 用 程序 中 ， 在 日 期 、 时 间 函 数 之 前 使 用 date_default timezone_setO 函 数 就 可 以 完成 对 
时 区 的 设置 。date_default_timezone set0) 函 数 的 语法 如 下 : 


用 php imi -记事 本 
文件 四 “ 蜗 回 加 格式 (DO) 查看 W， 帮助 ( 
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date_default timezone set(timezone): | 

参数 timezone 为 PHP 可 识别 的 时 区 名 称 ， 如 果 时 区 名 称 PHP 无 法 识别 ， 则 系统 采用 的 是 

UTC 时 区 。 | 
例如 ， 修 改 时 区 前 获取 到 的 时 间 如 图 1.72 所 示 ， 修 改 时 区 后 获取 到 的 时 间 如 图 1.73 所 示 。 | 会 内 


te [= EE 当 无 标题 文档 - icrosoft T2380 国 [+ 
| [| 
| 加 三- 加- 四 国 罗 | | 加 恶名 -站 国外 | “ 
[ENETZETTITR > 大 让 男 峰 wzr 00 yym/ 梧 国王 | 


Ee 于 可 人 
元 rm | | J | 
图 1.72 修改 时 区 前 获取 的 时 间 图 1.73 修改 时 区 后 获取 的 时 间 
获取 时 间 的 代码 如 下 : 
<2php 
echo date("Y-m-d H:i:s"); 
?> 
专家 点 评 


在 PHP 语言 中 ， 日 期 、 时 间 函 数 依赖 于 服务 器 的 地 区 设置 ， 而 PHP 默认 设置 的 是 标准 的 | 
格林 威 治 时 间 《〈 即 采用 的 是 零 时 区 )， 所 以 ， 如 果 用 户 没有 对 PHP 的 时 区 进行 设置 ， 那 么 使 用 | 
日 期 、 时 间 函 数 获取 的 将 是 英国 伦敦 本 地 时 间 〈 即 零 时 区 的 时 间 )。 | 

这 也 就 是 为 什么 要 对 PHP 的 时 区 进行 设置 的 原因 。 例 如 ， 以 东 八 区 为 例 ， 如 果 当 地 使 用 | 
的 是 北京 时 间 ， 那 么 如 果 没 有 对 PHP 的 时 区 进行 设置 ， 获 取 的 时 间 就 将 比 当地 的 北京 时 间 少 8 | 

个 小 时 。 


问题 18 ”如 何在 Dreamweaver 中 创建 站 点 ? 


问题 阐述 
如 何 通过 Dreamweaver 开发 工具 创建 一 个 站 点 呢 ? 
专家 解答 | 
在 Dreamweaver 开发 工具 中 ， 如 果 创建 了 站 点 ， 就 不 必 在 正 浏览 器 中 输入 地 址 ， 而 只 需 | 
直接 按 下 键盘 中 的 F12 键 ， 即 可 完成 对 所 创建 程序 的 浏览 操作 。 | 
在 Dreamweaver 中 创建 站 点 和 配置 测试 服务 器 时 , 一 定 要 注意 将 本 地 的 HITP 地 址 与 测试 | 
服务 器 中 的 URL 前 级 统一 ， 即 都 指定 到 站 点 的 根 目录 下 。 例 如 ， 指 定 HTTP 地 址 是 | 
“http://localhost//MR/Instance/”， 那 么 测试 服务 器 的 URL 前 组 也 必须 是 “http://localhost/MR/ | 
Instance/” 或 “http://127.0.0.1//MR/Instance/”。 | 
在 Dreamweaver 中 创建 站 点 的 操作 步 又 如 下 : | 
(1) 打开 Dreamweaver 开发 工具 ， 选 择 菜单 栏 中 的 “站 点 ”一 “新 建站 点 ”命令 ， 在 弹 | 
出 的 对 话 框 中 输入 站 点 名 称 ， 如 图 1.74 所 示 。 | 


ye L_ 
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(2) 单 击 “ 高 级 ”按钮 ， 将 弹出 如 图 1.75 所 示 的 对 话 框 。 在 该 对 话 框 中 设置 本 地 根 文件 
| 夹 ， 链 接 相对 于 “站 点 根 目录 ”， 并 且 设置 HTTP 地 址 。 
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EE x | 
本 
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rene 二 srssnweevsr 中 隐 中 点 昨 娘 件 和 文件 丙 的 集合 ， 它 对 应 二 限 务 著 上 的 8:b 7 自动 新 本 地 立 梓 光志 00) 


ed 歌 认 图 基文 件 赤 区 )- 百 
了 放 委 相 时 于 : 6“ 文 蔡 包 ) 个 站 点 根 目录 名) 
| 示例: Nysite 0TF 地 址 四: k 汪 
| i rr Hi Om i 
| 示例: Nt :Wwww mylpst emySite 区 分 灰 小 写 的 镍 按 : 厂 使 用 区 分 大 小 写 的 棒 授 检查 |) 
| 可 a ee 
| 5 和 殿后 a 
| 图 1.74 定义 站 点 图 1.75 定义 mr 站 点 


(3) 单 击 如 图 1.75 所 示 对 话 框 左 侧 的 “测试 服务 器 ”， 弹 出 如 图 1.76 所 示 的 测试 服务 器 
对 话 框 , 选择 服务 器 模型 : PHP MySQL, 访问 : 本 地 /网 络 , 测试 服务 器 文件 夹 : E:\AppServiwwww\ 
MR\Instance\，URL 前 级 : http://localhost/mr/Instance/， 最 后 单 击 “ 确 定 ” 按 钮 。 


a 的 站 点 定 文 为 


EE 
旬 黄 而 了 耻 务 办 
i 服务器 模 到 仙 s|3P SOL 二 
证 各 放 本 月 [不 地 /所 了 | 
ena Te 


Ca] mw RD 


图 1.76 配置 测试 服务 器 
| (4) mr 站 点 和 测试 服务 器 设置 完毕 ， 在 Dreamweaver 下 直接 使 用 快捷 键 F12 即 可 浏览 


| 程序 。 
| 专家 点 评 


在 Dreamweaver 中 创建 站 点 和 配置 测试 服务 器 时 , 如 果 本 地 的 HITP 地 址 与 测试 服务 器 中 
| 的 URL 前 缀 不 统一 ， 那 么 就 无 法 通过 F12 键 直接 浏览 程序 。 
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第 1] 章 PHP 开 发 规范 与 入 门 要 点 


问题 19 ”如何 使 用 Dreamweaver 开发 PHP 程序 ? 


问题 阐述 


Dreamweaver 是 Adobe 公司 的 Web 站 点 和 应 用 程序 的 专业 开发 工具 ， 它 将 可 视 布局 工具 、 


应 用 程序 开发 功能 和 代码 编辑 组 合 在 一 起 。 其 功能 强大 ， 使 得 各 个 层次 的 设计 人 员 和 开发 人 员 | 
都 能 够 美化 网 站 及 创建 应 用 程序 。 从 基于 CSS 设计 的 领先 支持 手工 编码 ，Dreamweaver 为 专业 | 
人 员 提 供 了 一 个 集成 、 高 效 的 环境 ， 这 样 开发 人 员 就 可 以 使 用 Dreamweaver 及 所 选择 的 服务 器 | 
来 创建 功能 强大 的 Intermet 应 用 程序 ， 从 而 使 用 户 能 连接 到 数据 库 、Web 服务 和 旧式 系统 。 那 | 


么 该 如 何 使 用 Dreamweaver 建立 站 点 及 开发 PHP 程序 呢 ? 
专家 解答 
利用 Dreamweaver 建立 站 点 及 开发 PHP 程序 的 具体 步骤 如 下 : 
(1) 在 Dreamweaver 下 建立 站 点 。 首 先 应 创建 一 个 文件 夹 ， 作 为 站 点 的 根 目录 。 
(2) 打开 Dreamweaver， 选 择 “ 站 点 ”一 “新 建站 点 ”命令 ， 如 图 1.77 所 示 。 
ROOT 一 


加 | 器 多 同 加 | 加 加 | 加 


© DREAMWEAVER 


打开 最 近 项 目 
wertp 

blioce nuaoap_baso pnp 
unto php 

司 ozsendpnn 


图 1.77 选择 新 建站 点 


(3) 打开 “新 建站 点 ”窗口 ， 如 图 1.78 所 示 。 在 该 窗口 中 选择 “基本 ”选项 不， 在 “您 
打算 为 您 的 站 点 起 什么 名 字 ” 文 本 框 中 输入 要 创建 的 站 点 名 称 ; 在 “您 的 站 点 的 HITP 地 址 | 


(URL) 是 什么 ? ”文本 框 中 输入 运行 时 的 URL 地 址 ， 定 位 到 创建 的 根 目 录 下 。 


(4) 选择 图 1.78 中 的 “高 级 ”选项 卡 ， 打 开 如 图 1.79 所 示 的 窗口 ， 其 中 的 “站 点 名 称 ” | 
文本 框 中 的 内 容 自 动 更 改 为 在 “基本 ”选项 卡 中 为 站 点 所 取 的 名 称 “01”。 在 “本 地 根 文件 夹 ”| 
文本 框 中 输入 或 选择 在 步骤 1) 中 建立 的 网 站 根 文件 夹 “FMAppServiwww\mr\O1\”。 单 击 “ 确 | 


定 ” 按 钮 返回 “站 点 管理 ”窗口 ， 再 单 击 该 窗口 的 “完成 ”按钮 ， 成 功 创建 站 点 。 


(5) 选择 “文件 ”一 “新 建 ” 命 令 ， 将 弹出 如 图 1.80 所 示 的 对 话 框 。 在 该 对 话 框 的 类 别 


列表 中 选择 “动态 页 ”选项 ， 然 后 在 动态 页 列表 中 选择 PHP 选项 ， 单 击 “ 创 建 ”按钮 ， 在 | 
Dreamweaver 中 创建 PHP 文件 ， 在 该 文件 中 输入 如 下 代码 : 


本 UD 


一 a 人 Pp 六 须知 道 的 300 个 癌 是 


1.78 “01 的 站 点 定义 为 ”窗口 中 的 图 1.79 “01 的 站 点 定义 为 ”窗口 中 的 
“基本 ”选项 卡 “高 级 ”选项 卡 


1.80 新 建 PHP 文件 


(6) 将 该 文件 命名 为 index.php， 保 存在 创建 的 站 点 的 根 目录 下 ， 如 图 1.81 所 示 。 
(7) 打开 浏览 器 ， 在 地 址 栏 中 输入 http://192.168.1.59/mr/01/005/sl/index.php， 如 果 浏 览 器 
中 输出 如 图 1.82 所 示 的 页 面 ， 则 说 明 利 用 Dreamweaver 开发 PHP 程序 得 以 实现 。 


! 1 [ERRORETEETCEEYT 


| 图 1.81 文件 的 保存 图 1.82 程序 运行 结果 
| 
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第 1] 章 PHP 开发 规范 与 入 门 要 点 


专家 点 评 


在 通过 Dreamweaver 开发 PHP 程序 的 过 程 中 , 如果 想 要 在 Dreamweaver 中 使 用 快捷 键 F12 
来 浏览 程序 , 那么 必须 要 在 创建 站 点 时 对 服务 器 进行 配置 , 否则 将 不 能 使 用 快捷 键 来 浏览 程序 。 | 


问题 20 ”如 何 使 用 Dreamweaver 识别 .phtml 的 模板 文件 ? 


问题 阐述 
如 何 使 用 Dreamweaver 识别 .phtml 格式 的 模板 文件 呢 ? 
专家 解答 
解决 方法 如 下 : 
(1) 找到 Dreamweaver 安装 目录 下 的 Configuration\DocumentTypes\MMDocumentTypes 
.Xml 文件 并 将 其 打开 。 
(2) 将 MMDocumentTypes.xml 文件 中 的 如 下 代码 : 
<documenttype id="PHP MySQL" servermodel="PHP MySQL" internaltype="Dynamic" winfilee- | 
xtension="php,php3,php4,php5" macfileextension="php,php3,php4,php5" file="Default.php" writebyteordermark= | 
"false"> | 
改 为 时 | 
<documenttype id="PHP_MySQL" servermodel="PHP MySQL" intemaltype= "Dynamic" winfileextension= | 
"php.php3,php4,php5,phtml" macfileextension="php.php3.php4.php5,phtml" file="Defaultphp" writebyteordermark= | 
a | 
(3) 启动 Dreamweaver 可 以 发 现 ， 以 .phtml 为 扩展 名 的 模板 文件 也 可 以 在 其 可 视 编辑 模 | 
式 下 进行 编辑 开发 。 | 
例如 ， 使 用 Dreamweaver 编辑 index.phtml 文件 ， 如 图 1.83 所 示 。 


图 1.83 ”使 用 Dreamweaver 编辑 index.phtml 文件 


专家 点 评 | 

PHTML (有 时 也 称 为 PHP) 网 页 是 一 种 包含 PHP (一 种 和 JavaScript 或 Microsoft VBScript | 

类 似 的 语言 ) 脚本 的 网 页 ， 和 ASP 一 样 ，PHP 脚本 也 是 镶嵌 在 网 页 的 HTML 代码 之 中 的 。 在 | 

页 面 被 发 送 给 请 求 的 用 户 之 前 ， 网 页 服务 器 调用 PHP 解释 程序 来 解释 和 执行 PHP 脚本 。 含 有 | 
Se We 


3 人 Pip 六 须 知道 的 300 个 辣 是 


PHP 脚本 的 网 页 通常 都 以 “php”、“.php3” 或 “phtml” 作 为 后 级 。 和 ASP 一 样 ，PHP 也 可 以 
被 认为 是 一 种 “动态 网 页 ”。 


问题 21 如 何在 Windows 下 安装 和 配置 phpMyAdmin? 


问题 前 述 

如 何在 Windows 下 安装 和 配置 phhpMyAdmin? 
专家 解答 

1. 安装 phpMyAdmin 图 形 化 管理 工具 

将 下 载 的 “phpMyAdmin-2.11.5-all-languages.rar” 文 件 解 压缩 到 Apache 主 目录 下 ,解压 后 
的 名 称 是 “phpMyAdmin-2.11.5-all-languages”， 其 中 2.11.5 是 pppMyAdmin 的 版 本 号 ， 为 了 便 
于 使 用 可 以 将 解压 缩 后 的 文件 夹 重 新 命名 为 pppMyAdmin。 

2. 配置 phpMyAdmin 图 形 化 管理 工具 

将 phppMyAdmin 安装 目录 下 的 libraries 目录 下 的 config.defaultphp 复制 到 phpmyAdmin 根 
目录 下 ， 并 改名 为 config.inc.php。 然 后 用 Dreamweaver 开发 工具 打开 config.inc.php 文件 ， 将 
光标 定位 到 指定 的 标识 行 位 置 。config.inc.php 文件 初始 值 设置 如 图 1.84 所 示 , 配置 后 的 文件 如 
图 1.85 所 示 。 


$cfel’ PaaAbsoluteUri] = *" $cfe[’ Paahbsolutel) wtp://localhost/phpnyadnin’ ;| 


图 1.84 配置 configinc.php 文件 前 图 1.85 配置 configinc.php 文件 后 


第 39 行 用 于 设置 pbpMyAdmin 网 址 。 

第 64 行 用 于 设置 访问 方式 为 cookie， 本 机 不 需要 设置 ， 但 是 网 络 需要 设置 成 cookie。 

第 151 行 用 于 设置 phpMyAdmin 的 登录 方式 。 可 以 设置 的 登录 方式 有 3 种 : config、http 
和 cookie。 其 中 config 是 默认 的 登录 方式 ， 该 方式 表示 用 户 正确 地 输入 用 户 名 和 密码 后 可 直接 
登录 ; 如果 使 用 http 方式 登录 ， 会 弹出 一 个 对 话 框 ， 提 示 输 入 用 户 名 和 密码 ; 如 果 使 用 cookie 
方式 登录 ,在 访问 phpMyAdmin 时 会 打开 一 个 登录 页 面 ， 提 示 用 户 输入 用 户 名 和 密码 ， 并 将 用 
户 名 和 密码 保存 在 系统 的 cookie 中 , 下 次 访问 时 用 户 名 和 密码 将 会 自动 显示 在 文本 框 和 密码 输 
入 框 中 ， 通 常用 于 互联 网 。 

第 158 行 用 于 设置 MySQL 数据 库 的 用 户 名 。 

第 165 行 用 于 设置 MySQL 数据 库 的 密码 。 

这 里 设置 的 用 户 名 和 密码 是 数据 库 中 已 经 存在 的 用 户 名 和 密码 , 如 果 设 置 的 用 户 名 和 密码 
在 数据 库 中 不 存在 ， 那 么 将 不 可 以 使 用 。 


说 明 : 
如 果 读 者 未 使 用 Dreamweaver 工具 打开 configinc.php 文件， 那么 对 应 的 行 号 会 有 所 差异 ， 
读者 可 以 搜索 相关 的 关键 宇 ， 如 , “[auth type]” 进 行 搜索 定位 ， 然 后 更 改 对 应 的 配置 文件 。 


。40 。 


设置 完毕 后 ， 保 存 configphp 文件 ， 在 浏览 器 中 输入 “http-/localhosphpMyAdmin”， 将 弹 
出 用 户 登 录 对 话 框 ， 如 图 1.86 所 示 。 | 


图 1.86 用 户 登 录 对 话 杠 


此 时 输入 正确 的 用 户 名 和 密码 〈 笔 者 在 安装 MySQL 时 ， 设 置 MySQL 数据 库 的 用 户 名 是 
“root”， 密 码 是 “root”)， 即 可 进入 phpMyAdmin 的 主页 面 。 


问题 22 ”如 何在 Linux 下 安装 和 配置 phpMyAdmin? 


问题 阐述 
如 何在 Linux 下 安装 和 配置 phppMyAdmin? 
专家 解答 
1， 安装 phpMyAdmin 图 形 化 管理 工具 
在 Linux 操作 系统 下 安装 phpMyAdmin， 将 下 载 的 软件 包 “phpMyAdmin-2.11.5-al - | 
languagesrar” 解 压缩 到 配置 环境 的 根 目录 下 ， 为 了 便于 使 用 ， 可 以 将 解压 缩 后 的 文件 夹 重 新 | 
命名 为 phpMyAdmin。 | 
命令 模式 下 的 操作 如 下 : 


#1mv phpMyAdmin-2.11.5-all-languages.rar/var/ww/html 
#tar zvxf phpMyAdmin-2.11.5-all-languages.rar 
#mv phpMyAdmin-2.11.5-all-languages phpMyAdmin 


通过 上 述 3 个 命令 将 下 载 的 软件 包 解压 缩 到 /varhwww/html/ 目 录 下 , 从 而 完成 phpMyAdmin | 
图 形 化 管理 工具 的 安装 。 | 
2. 配置 phpMyAdmin 图 形 化 管理 工具 
配置 在 phpMyAdmin 目录 下 的 config.inc.php 文件 下 进行 ， 更改 如 下 参数 的 配置 。 


// 将 参数 PmaAbsoluteUri 的 值 设 定 为 phppMyAdmin 所 处 的 位 置 : 
$cfg[PmaAbsoluteUri] = 'http://localhost/phpMyAdmin'; 


S$cfe['blowfish secret']='cookie': // 设 定 访问 方式 为 Cookie 
$cfg['Servers'][$il[host] = "localhost'; // 设 定 MySQL 所 在 的 主机 名 或 他 地址 
$cfg['Servers'][$il[auth type] = ‘http'; // 设 定 进 入 phpMyAdmin 管理 MySQL 的 方式 
$cfg['Servers'][$il[mser] = root: // 设 定 管理 MySQL 的 账号 
$cfg['Servers'][$il[password'] = Toot: // 设 定 管理 MySQL 的 密码 


参数 设置 完成 后 保存 该 文件 ， 重 新 启动 Apache 服务 器 。 | 
Ey L_ 


人 ip 六 预 知道 的 300 个 问题 


在 浏览 器 的 地 址 栏 中 输入 “http:/localhostphpMyAdmin/” 即 可 打开 phpMyAdmin 图 形 化 


| 管理 主页 面 进行 数据 库 操作 。 
内 | 问题 23 如何 应 用 phpMyAdmin 修改 数据 库 的 
用 户 名 和 密码 ? 
| 问题 阐述 


| 修改 MySQL 数据 库 的 用 户 名 和 密码 的 方法 有 很 多 ， 应 用 phpMyAdmin 来 修改 是 十 分 方便 
| 快捷 的 方法 ， 那 么 如 何 操作 呢 ? 

| 专家 解答 

| 通过 phpMyAdmin 图 形 化 管理 工具 来 修改 数据 库 用 户 名 和 密码 的 具体 操作 如 下 。 

| (1) 单 击 phppMyAdmin 主页 面 中 的 旦 权限 超 链接 ， 打 开 服 务 器 用 户 操作 页 面 ， 如 图 1.87 
| 所 示 。 在 该 页 面 中 ， 可 以 对 指定 用 户 的 权限 进行 编辑 ， 可 以 添加 新 用 户 和 删除 指定 的 用 户 。 
| 这 里 选择 指定 的 用 户 ， 单 击 鲍 (编辑 权限 ) 超 链接 ， 对 指定 用 户 的 权限 进行 设置 ， 进 入 如 
| 图 1.88 所 示 的 页 面 。 
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图 1.87 服务 器 用 户 一 览 表 图 1.88 编辑 用 户 权限 

| (2) 在 图 1.88 所 示 的 页 面 中 ， 可 以 设置 用 户 的 权限 、 修 改 密码 、 更 改 登录 用 户 的 信息 和 
| 复制 用 户 。 在 输入 原 密码 和 新 密码 之 后 ， 单 击 “ 执 行 ”按钮 即 可 完成 对 用 户 密码 的 修改 操作 ， 
| 返回 主页 面 ， 如 图 1.89 所 示 。 


le /localhest_| phpspkinin 2 LE nx 
Er TT E32 
OSE -OO- dn ra 
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root 加 Iacalhost 的 客 码 已 成 功 更 下. 


三 SQL 查询 
SETPASSWORD FOR Yat ocahost = PASSNORD ~—) 


[编辑 ] [创建 PHP 代码 ] 


TT 玉 
| 图 1.89 用 户 密码 修改 成 功 
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第 1] 章 PHP 开 发 规范 与 入 门 要 点 


问题 24 ”如 何 应 用 phpMyAdmin 创建 数据 库 与 数据 表 ? 
问题 阐述 


数据 库 与 数据 表 的 创建 可 以 通过 命令 符 来 实现 ， 但 是 使 用 该 方法 对 于 初学 者 来 说 比较 麻 
烦 ， 而 且 因 其 需要 手写 的 代码 比较 多 ， 很 容易 出 现 错误 ， 那 么 如 何 才能 简单 而 直接 地 创建 数据 ， 
库 与 数据 表 呢 ? | 
专家 解答 
1， 创 建 数据 库 | 
在 phpMyAdmin 主页 面 的 “创建 一 个 新 的 数据 库 ”文本 框 中 输入 数据 库 的 名 称 “db。 | 
study”， 然 后 在 其 下 方 的 下 拉 列 表 框 中 选择 所 要 使 用 的 编码 ， 一 般 选 择 “gb2312_Chinese_ci” | 


简体 中 文 编码 格式 ， 单 击 “ 创 建 ” 按 钮 ， 创 建 数 据 库 ， 如 图 1.90 所 示 。 
数据 库 创建 成 功 后 ， 将 显示 如 图 1.91 所 示 的 页 面 。 


| TT 


图 1.90 phpMyAdmin 管理 主页 面 图 1.91 数据 库 创 建成 功 


说 明 : | 
在 图 1.91 所 示 页 面 的 右 侧 区 域 中 可 以 对 该 数据 库 进行 相关 操作 ， 如 结构 、SQL、 导 出 、 搜 | 
索 、 查 询 、 删 除 等 ， 单 击 相应 的 超 链接 即 可 进入 相应 的 操作 页 面 。 但 是 在 创建 的 数据 库 还 没有 | 
创建 数据 表 的 情况 下 ， 只 能 够 执行 结构 、SQL、Import、 操 作 、 权 限 和 删除 6 项 操作 ， 而 户 搜索 、 | 
唱 查 询 和 部 导出 这 3 项 操作 不 能 执行 ， 当 光标 指向 这 几 个 超 链接 时 ， 将 弹出 不 可 用 标记 加 . | 


2. 创建 数据 表 


数据 库 创 建成 功 后 ， 在 该 数据 库 下 创建 tb_user 数据 表 。 在 db_study 数据 库 右 侧 的 操作 页 
面 中 和 输入 数据 表 的 名 称 和 字段 数 ， 然 后 单 击 “ 执 行 ”按钮 ， 即 可 创建 数据 表 ， 如 图 1.92 所 示 。 | 
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| 


图 1.92 ”创建 数据 表 
过 二 


再 人 ip 六 预 知道 的 300 个 问题 
成 功 创建 数据 表 tb_admin， 将 显示 数据 表 结 构 页 面 。 在 表单 中 对 各 个 字段 的 详细 信息 进行 
| 录入 ， 包括 字段 名 、 数 据 类 型 、 长 度 / 值 、 编 码 格式 、 是 否 为 空 、 主 键 等 ， 以 完成 对 表 结 构 的 详 
| ， 细 设置 。 当 所 有 的 信息 都 输入 完成 以 后 ， 单 击 “保存” 按钮 ,创建 数据 表 结 构 ， 如 图 1.93 所 示 
人 A 
E | 在 定义 数据 表 的 格式 时 必须 指定 主键 ， 其 类 型 为 int 
ee 
2 | 数据 表 创 建成 功 ， 效 果 如 图 1.94 所 示 


ET 加 雪 锯 库 : db_smudy ， 回 表 :tb_user 


i 攀 天 SQL 广 执 雪 于: 医科 出 _ 吕 npor 各 如 企 _ 甸 沛 字 天 


国 和 “0_shydy tb User 已 经 建立 。 
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图 1.93 创建 数据 表 结 构 


图 1.94 tb_user 数据 表 


问题 25 如 何 应 用 phpMyAdmin 导入 和 


导出 MySQL 数据 库 脚 本 ? 
问题 六 述 


phpMyAdmin 是 一 个 使 用 PHP 编写 的 、 基 于 Web 的 MySQL 管理 工具 ， 可 以 通过 互联 网 
控制 和 操作 MySQL。 那 么 该 如 何 应 用 phpMyAdmin 导入 和 导出 MySQL 数据 库 脚本 呢 ? 
专家 解答 


导入 和 导出 MySQL 数据 库 脚本 是 互 逆 的 两 个 操作 。 导 入 是 执行 扩展 名 为 “.sql” 的 文件 ， 
| 将 数据 导入 数据 库 中 ;导出 是 将 数据 表 结 构 、 表 记录 存储 为 

| 导出 的 操作 实现 数据 库 的 备份 和 还 原 。 

| 1. 导出 MySQL 数据 库 脚本 
单 击 phppMyAdmin 主页 面 中 的 圈 号 出 超 链接 ， 打 开导 出 编辑 区 ， 如 图 1.95 所 示 。 选 择 导 
| 出 文件 的 格式 ， 这 里 默认 使 用 选项 “SQL”， 选 中 “另存 为 文件 ” 复 选 框 ， 单 击 “ 执 行 ”按钮 
| 弹出 如 图 1.96 所 示 的 文件 下 载 对 话 框 ， 单 击 “ 保 存 ” 按 钮 ， 将 脚本 文件 以 “ 
| 指定 位 置 。 


“.sql” 的 脚本 文件 。 通 过 导入 和 


.sql” 格 式 存储 在 


。44 。 
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《到 记 信人 要 ) 


天 要 打开 区 什 是 格 它 保 作证 生机 
EN 打开 外 取消 | 溢 四 信息 
中 单 击 “ 执 行 ”按钮 ， 即 可 生成 脚本 于 


图 1.95 生成 MySQL 脚本 文件 设置 页 面 图 1.96 存储 MySQL 脚本 对 话 框 
2. 导入 MySQL 数据 库 脚本 
单 击 欧 Import 超 链接 ， 进 入 执行 MySQL 数据 库 脚本 页 面 ， 单 击 “ 浏 览 ”按钮 查找 脚本 文 | 
件 (如 db_study.sql) 所 在 位 置 ， 如 图 1.97 所 示 ， 单 击 “ 执 行 ”按钮 ， 即 可 执行 MySQL 数据 库 | 
脚本 文件 。 | 


困 服务 黑 : bcalhost 》 品 笋 氢 库 ub_stedy 》 园 表 :由 aq Py | 
_ 转 刘 改 。 疝 关 构 好 SoL 万 得 二 了: 括 入 el en 有 国 二 于 ， 较 开 给 
Import 


文本 文件 的 位 置 【Fndatadb_siuqy sql 。“ 浏 距 | (最 大 限制: 2045 KB 
FT 习 


人 


Format of mported fie- 
C csy 


Options- 
C cayusingLohD DATA | EDLcomoatplt mode 
NONE 了 


@@ 单 击 “ 执 行 ” 按钮 即 可 生效 卢 画 


[a [iT mew 


图 1.97 执行 MySQL 数据 库 脚本 文件 


Rl 


专家 点 评 


在 执行 导入 操作 之 前 ， 首 先 要 检测 数据 库 中 是 否 存 在 与 所 导入 数据 库 同名 的 数据 库 ， 如 果 
不 存在 ， 则 首先 要 在 数据 库 中 创建 一 个 名 称 与 数据 文件 中 的 数据 库 名 相同 的 数据 库 ， 然 后 再 执 
。45 。 


Pip 六 预 知道 的 300 个 问题 


LT 
外 多 
| 行 MySQL 数据 库 脚本 文件 。 另 外 ， 在 当前 数据 库 中 ， 不 能 有 与 将 要 导入 数据 库 中 的 数据 表 重 
| 名 的 数据 表 存在 ， 如 果 存在 则 导入 文件 就 会 失败 ， 并 提示 错误 信息 。 

| 另外, 读者 也 可 通过 单 击 phppMyAdmin 图 形 化 工具 左 侧 区 的 国 按 钮 , 在 打开 的 对 话 框 中 音 
| 击 “ 导 入 文件 ” 超 链接 ， 然 后 选择 脚本 文件 所 在 的 位 置 ， 从 而 执行 数据 导入 操作 。 


人 | 
问题 26 如何 打开 magic quotes gpc 来 防止 SQL 注入 ? 


| 问题 阐述 
| SQL 注入 是 非常 危险 的 事情 ， 轻 则 网 站 后 台 被 入 侵 ， 重 则 整个 服务 器 沦陷 ， 所 以 用 户 一 定 
| 要 小 心 。 那 么 该 如 何 防止 SQL 的 注入 呢 ? 
| 专家 解答 
| php.ini 中 有 一 个 设置 : 

magic_quotes_gpc = Of 
| 该 设置 默认 是 关闭 的 ， 如 果 它 打开 后 将 自动 把 用 户 提 交 对 sql 的 查询 进行 转换 ,例如 把 “” 
| 转换 为 “\” 等 ， 这 对 防止 sql 注入 有 重大 作用 。 所 以 我 们 将 magic_quotes_gpc 设置 为 On。 设 
| 置 代码 如 下 : 
| magic_quotes gpc = On 


问题 27 ”如 何 对 错误 提示 信息 进行 控制 ? 


| 问题 阐述 
| 一 般 php 在 没有 连接 到 数据 库 或 其 他 情况 下 时 会 有 错误 提示 ， 通 常 错误 信息 中 会 包含 php 
| 脚本 当前 的 路 径 信息 或 查询 的 SQL 语句 等 信息 ， 这 类 信息 提供 给 黑客 后 是 不 安全 的 ， 所 以 一 
| 般 服务 器 建议 禁止 错误 提示 。 那 么 如 何 才能 有 效 地 控制 错误 信息 提示 呢 ? 
| 专家 解答 
| 在 phpini 文 件 中 找到 
display_errors = On 
将 其 改 为 : 
display_errors = Of 
即 可 将 错误 提示 信息 屏蔽 掉 。 
如 果 一 定 要 显示 错误 信息 ， 可 以 设置 显示 错误 的 级 别 ， 如 只 显示 警告 以 下 的 信息 : 
error reporting =E WARNING & E _ ERROR 
| 当然 ， 错 误 提示 关闭 后 ， 很 难 找到 错误 原因 ， 所 以 ， 有 需要 时 可 以 暂时 开启 ， 不 需要 时 再 
| 将 其 关闭 。 
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问题 28 ”如 何 记录 错误 日 志 ? 


问题 阐述 
为 了 方便 查找 服务 器 的 运行 原因 ， 最 好 对 错误 日 志 进 行 记录 ， 那 么 该 如 何 记 录 日 志 呢 ? 
专家 解答 


1. 设置 错误 日 志 存储 位 置 
在 php.ini 文件 中 ， 关 闭 display_errors 后 把 错误 信息 记录 下 来 : 

log errors = On 
同时 也 要 设置 错误 日 志 存 放 的 目录 ， 建 议和 Apache 的 日 志 存 在 一 起 : 

error_log = D:/usr/local/apache2/logs/php_error.log 
Apache 用 户 必须 具备 错误 日 志文 件 的 写 权限 ， 否 则 不 能 将 错误 信息 添加 到 日 志文 件 中 。 | 
2. 降低 用 户 权限 
在 MySQL 服务 器 中 ， 默 认 使 用 的 登录 用 户 是 本 地 系统 用 户 ， 该 用 户 的 权限 非常 大 ， 为 了 | 

控制 登录 MySQL 服务 器 用 户 的 权限 ， 可 以 通过 创建 新 用 户 的 方式 来 降低 用 户 的 权限 。 | 

其 方法 是 应 用 net user 命令 创建 新 用 户 ， 并 且 将 该 用 户 从 组 中 删除 。 其 命令 如 下 : 


net user mr 123456/add 
net localgroup users mr /del 
创建 名 称 为 mr 的 用 户 , 密码 为 123456, 并 且 将 其 从 组 [ne 一 ES 
中 删除 。 于 [| 
新 的 标准 用 户 添加 成 功 后 ， 通 过 该 新 用 户 来 登录 | mseo ss 
MySQL 服务 器 , 这 样 就 降低 了 用 户 的 权限 。 其 具体 方法 是 : SAD -本 


在 系统 的 “服务 ”中 定位 到 MySQL， 右 击 MySQL， 0 
在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ， 在 弹出 的 对 话 框 “| aaamamesesass 
中 选择 “登录 ”选项 卡 ， 如 图 1.98 所 示 。 区 
在 “登录 身份 ”选项 组 中 选中 “此 账户 ” 单 选 按钮 ， | 天 
找到 所 创建 的 用 户 ， 并 添加 用 户 密码 ， 单 击 “确定 ” 按钮。 图 198 设置 MySQL 的 各 录用 记 
重新 启动 MySQL 服务 ，MySQL 即 运行 在 低 权限 下 。 
在 Windows 操作 系统 下 ，Apache 默认 运行 的 是 system 权限 ， 可 以 应 用 上 述 的 方法 来 降低 | 
Apache 的 权限 。 | 


专家 点 评 


这 里 还 可 以 通过 设置 各 个 文件 夹 的 权限 ， 来 让 Apache 用 户 只 能 执行 设 定 的 事情 ， 为 每 一 
个 目录 建立 一 个 单独 能 读 写 的 用 户 。 这 也 是 当前 很 多 虚拟 主机 提供 商 的 流行 配置 方法 。 


A L_ 


如 Pp 六 须 知道 的 300 个 辣 是 


问题 29 ”如 何 解决 使 用 MySQL 函数 导致 


全] Apache(IIS) 岗 溃 的 问题 ? 


问题 请 述 


| 本 问题 中 所 说 的 崩溃 是 在 Windows 下 配置 PHP 环境 时 发 生 的 。 具 体 的 表现 是 php 运行 正 
| 常 ， 但 是 只 要 一 调用 MySQL 的 相关 函数 ，Apache 就 会 崩溃 。 

环境 配置 方式 : 

php 以 模块 的 方式 运行 在 Apache 下 。 运行 phpinfo0) 函 数 ， 显 示 mysql 库 加 载 成 功 , 但 调用 
| MySQL 函数 时 则 Apache 骨 溃 ， 在 Windows 的 错误 日 志 中 记录 以 下 内 容 : 
| 错误 应 用 程序 httpd.exe， 版 本 2.2.11.0， 错 误 模块 php5ts.dl， 版 本 5.2.10.10， 错 误 地 址 
| 0x0000acGa。 


| 专家 解答 

| 在 安装 MySQL 时， 在 配置 设 定 时 把 MySQL 的 Bin 目录 放 到 了 系统 环境 变量 Path 中 。 而 
| Bin 目录 中 正好 有 一 个 libmysql.dll 的 库 。 

| 而 PHP 在 加 载 MySQL 库 时 ， 会 搜索 系统 环境 变量 Path 的 每 一 个 目录 寻找 libmysqldll， 

| 由 于 MySQL 的 bin 目录 也 在 其 中 ， 那 么 PHP 就 先 把 MySQL 的 libmysql.dll 加 载 了 ， 所 以 就 会 
， 由 于 库 的 兼容 问题 导致 MySQL 的 函数 无 法 运行 。 

| 解决 方案 : 


| 将 MySQL 的 bin 目录 从 系统 环境 变量 Path 中 移出 ， 或 者 将 libmysql.dll 从 MySQL 的 bin 
| 目录 中 移出 ， 并 且 重 新 启动 系统 。 


问题 30 HTTP Apache 错误 代码 都 有 哪些 ? 
分 别 代表 什么 含义 ? 


| 问题 阐述 
| 在 运行 PHP 程序 时 ， 如 果 发 生 错误 ， 都 会 返回 一 个 错误 代码 ， 准 确 地 解读 这 个 错误 代码 
| 的 含义 ， 对 解决 程序 中 的 问题 非常 关键 ， 有 了 它 就 可 以 准确 地 定位 错误 出 现 的 位 置 。 那 么 如 何 
| 才能 准确 地 解读 错误 代码 呢 ? 
| 专家 解答 
| 为 了 帮助 读者 更 好 地 解读 HTPP Apache 中 的 错误 代码 ,特此 对 错误 代码 进行 了 汇总 ， 让 读 
| 者 在 学 习 过 程 中 有 一 个 参考 。HTTP 状态 码 基本 上 可 以 分 为 5 类 : 
1。1xx 为 消息 类 ， 该 类 状态 代码 用 于 表示 服务 器 临时 回应 
100 Continue 表示 初始 的 请 求 已 经 被 服务 器 接受 ， 浏 览 器 应 当 继续 发 送 请 求 的 其 余部 分 
(HTTP 1.1)。 
2 。48 。 
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101 Switching Protocols 服务 器 将 遵从 客户 的 请 求 转换 到 另外 一 种 协议 CHITP 1.1)。 
2. 2xx 表示 浏览 器 端 请 求 被 处 理 成 功 
表 1.3 2xx 浏览 器 端 请 求 处 理 代码 说 明 


代 码 说 明 
200 OK - 切 正常 
201 Created 服务 器 已 经 创建 了 文档 ，Location 头 给 出 了 它 的 URL | 
202 Accepted 已 经 接受 请 求 ， 但 处 理 尚未 完 局 | 


203 Non-Authoritative | 文档 已 经 正常 地 返回 ， 但 一 些 应 答 头 可 能 不 正确 ， 因 为 使 用 的 是 文档 的 拷贝 | 
Information (HTTP 1.1 新 ) 

204 No Content 没有 新 文档 ， 浏 览 器 应 继续 显示 原来 的 文档 。 这 与 表 1.4 中 的 304 非 常 相 似 
没有 新 的 内 容 , 但 浏览 器 应 该 重 置 它 所 显示 的 内 容 。 用 来 强制 浏览 器 清除 表单 输 


205 Reset Content 入 内 容 ( 1.1 新 ) 
- 客户 发 送 了 一 个 带 有 Range 头 的 GET 请 求 ， 服 务 器 完成 了 它 CHITP 1.1 新 ) 。 
206 Partial Content 


注意 ， 通 过 Range 可 以 实现 断 点 续 传 
3，3XX 重 定向 
表 1.4 3xx 重 定向 代码 说 明 


代 码 说 阴 
2 客户 请 求 的 文档 可 以 在 多 个 位 置 找 到 ， 这 些 位 置 已 经 在 返回 的 文档 内 列 出 。 如 
300 Multiple Choices 


果 服 务 器 要 提出 优先 选择 ， 则 应 该 在 Location 应 答 头 指明 

客户 请 求 的 文档 在 其 他 地 方 ， 新 的 URL 在 Location 头 中 给 出 ， 浏 览 器 应 该 自动 地 
访问 新 的 URL 

类 似 于 301， 但 新 的 URL 应 被 视 为 临时 性 的 替代 ， 而 不 是 永久 性 的 。 注 意 ， 在 
HTTP1.0 中 对 应 的 状态 信息 是 “Moved Temporatily”。 出 现 该 状态 代码 时 ,浏览 
器 能 够 自动 访问 新 的 URL， 因 此 它 是 一 个 很 有 用 的 状态 代码 。 

注意 : 这 个 状态 代码 有 时 可 以 和 301 替 换 使 用 。 例 如 ， 如 果 浏 览 器 错误 地 请 求 
http:/host/~user〈 缺 少 了 后 面 的 斜 枉 )》 ， 有 的 服务 器 返回 301， 有 的 则 返回 302。 
严格 地 说 ， 我 们 只 能 假定 只 有 当 原来 的 请 求 是 GET 时 浏览 器 才 会 自动 重 定向 。 
可 参见 307 

类 似 于 301/302， 不 同 之 处 在 于 ， 如 果 原 来 的 请 求 是 POST，Location 头 指定 的 重 
定向 目标 文档 应 该 通过 GET 提 取 (HTTP 1.1 新 ) | 
客户 端 有 缓冲 的 文档 并 发 出 了 一 个 条 件 性 的 请 求 〈 一 般 是 提供 If-Modified-Since | 
头 表示 客户 只 想 比 指定 日 期 更 新 的 文档 ) 。 服 务 器 告诉 客户 ， 原 来 缓冲 的 文档 | 
还 可 以 继续 使 用 | 
客户 请 求 的 文档 应 该 通过 Location 头 所 指明 的 代理 服务 器 提取 (HTTP 1.1 新 ) | 
和 302(Found) 相 同 。 许 多 浏览 器 会 错误 地 响应 302 应 答 进行 重 定向 ， 即 使 原来 的 | 
请 求 是 POST， 即 使 它 实 际 上 只 能 在 POST 请 求 的 应 答 是 303 时 才能 重 定向 。 由 于 | 
这 个 原因 , HTTP 1.1 新 增 了 307， 以 便 更 加 清楚 地 区 分 几 个 状态 代码 : 当 出 现 303 
应 答 时 ,浏览 器 可 以 跟随 重 定向 的 GET 和 POST 请 求 ; 如 果 是 307 应 答 ， 则 浏览 器 
只 能 跟随 对 GET 请 求 的 重 定向 (HTTP 1.1 新 ) 


301 Moved Permanently 


302 Found 


303 See Other 


304 Not Modified 


305 Use Proxy 


307 Temporary Redirect 
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4.，4xX 错误 


表 1.5 4xx 错误 代码 说 明 


代 码 说 明 

400 Bad Request 请 求 出 现 语法 错误 
客户 试图 未 经 授权 访问 受 密码 保护 的 页 面 。 应 答 中 会 包含 一 个 

401 Unauthorized WWW-Authenticate 头 ， 浏 览 器 据 此 显示 用 户 名 字 / 密 码 对 话 框 ， 然 后 在 
填写 合适 的 Authorization 头 后 再 次 发 出 请 求 

i 资源 不 可 用 。 服务 器 理解 客户 的 请 求 ， 但 拒绝 处 理 它 。 通 常 由 于 服务 器 
上 文件 或 目录 的 权限 设置 导致 

404 Not Found 无 法 找到 指定 位 置 的 资源 。 这 也 是 一 个 常用 的 应 答 


请 求 方法 (GET、POST、HEAD、Delete、PUT、TRACE 等 ) 对 指定 的 


405 Method Not Allowed 资源 不 适用 ( i 
定 的 资源 已 经 找到 ， 但 它 前 类 型 ne 
406 Not Acceptable 指定 的 资源 已 经 找到 ， 但 它 的 MIME 类 型 和 客户 在 Accpet 头 中 所 指定 的 


不 兼容 (HTTP 1.1 新 ) 
407 Proxy Authentication Required | 类 似 于 401， 表 示 客 户 必 须 先 经 过 代理 服务 器 的 授权 CHTTP 1.1 新 ) 
在 服务 器 许可 的 等 待 时 间 内 ， 客 户 一 直 没 有 发 出 任何 请 求 。 客 户 可 以 在 


人 以 后 重复 同一 请 求 HTTP 1.1 新 ) 
A 通常 和 PUT 请 求 有 关 。 由 于 请 求 和 资源 的 当前 状态 相 冲 突 ， 因 此 请 求 不 
能 成 功 。 (HTTP 1.1 新 ) 
所 请 求 的 文档 已 经 不 再 可 用 ， 而 且 服 务 器 不 知道 应 该 重 定向 到 哪 一 个 地 
410 Gone 址 。 它 和 404 的 不 同 在 于 ， 返 回 407 表 示 文 档 永久 地 离开 了 指定 的 位 置 ， 
而 404 表 示 由 于 未 知 的 原因 文档 不 可 用 (HTTP 1.1 新 ) 
411 Length Required 服务 器 不 能 处 理 请 求 ， 除 非 客 户 发 送 一 个 Content-Length 头 HTTP 1.1 新 ) 
412 Precondition Failed 请 求 头 中 指定 的 一 些 前 提 条 件 失败 (HTTP 1.1 新 ) 


目标 文档 的 大 小 超过 服务 器 当前 愿意 处 理 的 大 小 。 如 果 服 务 器 认为 自己 
能 够 稍 后 再 处 理 该 请 求 ， 则 应 该 提供 一 个 Retry-After 头 (HITP 1.1 新 ) 
414 Request URI Too Long URI 太 长 (HTTP 1.1 新 ) 

416 Requested Range Not Satisfiable | 服务 器 不 能 满足 客户 在 请 求 中 指定 的 Range 头 (HTTP 1.1 新 ) 


5.，5xx 服务 器 错误 


413 Request Entity Too Large 


表 1.6 5xx 服务 器 错误 代码 说 明 


代码 说 了 明 
500 Internal Server Error 服务 器 遇 到 了 未 意料 到 的 情况 ， 不 能 完成 客户 的 请 求 
sO dt peranted 服务 器 不 支持 实现 请 求 所 需要 的 功能 。 例 如 ， 客 户 发 出 了 一 个 服务 器 不 
支持 的 PUT 请 求 
S02 Dad Glewiy 服务 器 作为 网 关 或 代理 时 ， 为 了 完成 请 求 访问 下 一 个 服务 器 ,但 该 服务 
器 返回 了 非法 的 应 答 


服务 器 由 于 维护 或 负载 过 重 未 能 应 答 。 例 如 ，Servlet 可 能 在 数据 库 连 接 
池 已 满 的 情况 下 返回 503。 服 务 器 返回 503 时 可 以 提供 一 个 Retry-After 头 


503 Service Unavailable 


由 作为 代理 或 网 关 的 服务 器 使 用 ， 表 示 不 能 及 时 地 从 远程 服务 器 获得 应 
答 (HTTP 1.1 新 ) 


504 Gateway Timeout 


服务 器 不 支持 请 求 中 所 指明 的 HTTP 版 本 (HTTP 1.1 新 ) 
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505 HITP Version Not Supported 
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问题 31 提高 编程 效率 的 24 招 

问题 32 ”为 什么 页 面 之 间 无 法 传递 变量 ? 
问题 33 为 什么 Win32 下 Apache2 用 get 
方法 传递 中 文 参数 会 出 错 ? 

问题 34 如 何 从 文本 中 移 除 空白 内 容 ? 
问题 33 如 何 获取 上 传 文件 的 后 缓 ? 
问题 36 如 何 实现 货币 数据 的 格式 化 
输出 ? 

问题 37 如 何 解决 用 Substr0 函 数 对 中 文 
字符 串 截 取 时 乱码 ? 

问题 38 ”如 何 获取 多 个 数组 元 素 的 键 名 ? 
问题 39 为 什么 $array[str] 错 了 ? 

问题 40 ”如 何 设 置 PHP 的 错误 级 别 ? 
问题 41 PHP 有 哪 几 种 注释 形式 ? 

问题 42 如 何 使 用 emptyO 函 数 判 断 文 本 
框 是 否 为 空 ? 

问题 43 如 何 使 用 strlen() 函 数 计算 文本 
长 度 ? 

问题 44 如何 使 用 copyO 函 数 对 文件 进 
行 复制 ? 

问题 45 如 何在 浏 
像 ? (fgetcO 函 数 ) 
问题 46 如 何 关闭 危险 函数 ? 
问题 47 如 何 将 数组 中 的 元 素 
符 串 ? 

问题 48 如 何 解决 使 用 empty 检查 函数 


览 器 中 直接 输出 图 


合成 字 


于 


章 


回 结果 时 报错 ? 
问题 49 如 何 使 用 mb _substr 函数 截取 中 
文字 符 串 ? 
问题 50 如 何 解决 PHP 无 法 执行 system() 
函数 的 问题 ? 
问题 S1 如 何 对 查询 关键 字 进 行 描 红 ? 
问题 52 ”如 何 对 字符 串 与 HTML 进行 转换 ? 
问题 53 如 何 屏蔽 违禁 词语 ? 
问题 $4 如 何 对 字符 串 进行 分 割 及 连接 ? 
问题 55 如 何 将 IP 地 址 转换 成 对 应 的 
数字 ? 
问题 56 
问题 57 
问题 58 
问题 59 
问题 60 
长 度 ? 
问题 61 如 何 定义 上 传 文件 的 名 称 ? 
问题 62 如 何 通过 字符 串 截 取 控 制图 像 
验证 码 的 输出 ? 
问题 63 如 何 打 乱 字符 串 中 字符 的 顺序 ? 
问题 64 如 何 对 字符 串 进 行 大 小 写 转换 ? 
问题 65 函数 strstr()、stristr()、strpos()、 
strrchr() 的 区 别 是 什么 ? 
问题 66 正和 switch 语句 有 何 区 别 ? 
问题 67 如 何 防 止 页 面 出 现 中 文 乱码 ? 


如 何 创 建新 型 字符 串 ? 

如 何 对 字符 串 进行 加 密 解密 ? 
如 何 比 较 字 符 串 的 大 小 ? 

如 何 获取 上 传 文件 的 扩展 名 ? 
如 何 控制 页 面 中 输出 字符 串 的 


人 ip 六 预 知道 的 300 个 问题 


KK% 
问题 31 提高 编程 效率 的 24 招 


问题 阐述 
在 开发 PHP 项 目的 过 程 中 ， 要 尽量 采用 标准 的 格式 书写 PHP 代码 ， 这 不 仅 是 作为 一 个 程 


| 序 员 的 良好 习惯 ， 同 时 还 能 够 提高 编程 效率 。 那 么 在 编程 过 程 中 ， 都 有 哪些 是 需要 注意 的 呢 ? 
| 专家 解答 


这 里 对 编程 中 需要 注意 的 一 些 规范 、 技 巧 进行 了 总 结 ， 供 读者 参考 ; 
(1) 用 单 引 号 代替 双 引号 来 包含 字符 串 ， 这 样 做 会 更 快 一 些 。 因 为 PHP 会 在 双 引 号 包围 


的 字符 串 中 搜寻 变量 ， 单 引号 则 不 会 。 注 意 : 只 有 echo 能 这 么 做 ， 它 是 一 种 可 以 把 多 个 字符 
| 串 当 作 参 数 的 “函数 ”( 说 明 : echo 是 语言 结构 ， 不 是 真正 的 函数 ， 故 把 函数 加 上 了 双 引 号 )。 


(2) $row['id'] 的 速度 要 比 Srow[id] 快 很 多 。 
(3) echo 比 print 快 ， 并 且 使 用 echo 的 多 重 参数 〈 说 明 : 指 用 逗号 而 不 是 句号 ) 代替 字 


符 串 连接 ， 比 如 echo $strl,$str2 。 


(4) foreach 效率 更 高 ， 尽 量 用 foreach 代替 while 和 for 循环 。 
(5) 在 执行 for 循环 之 前 确定 最 大 循环 数 ， 不 要 每 循环 一 次 都 计算 最 大 值 ， 最 好 运用 


| foreach 代替 。 


(6) 注销 那些 不 用 的 变量 尤其 是 大 数组 ， 以 便 释 放 内 存 。 
(7) 尽量 不 使 用 _get、_set、_autoload。 
(8) 使 用 include() 包 含 文件 时 ， 尽 量 使 用 绝对 路 径 ， 因 为 它 省 去 了 PHP 去 include path 


中 查找 文件 的 时 间 ， 解 析 操 作 系 统 路 径 所 需 的 时 间 会 更 少 。 


(9) 如 果 想 获取 脚本 开始 执行 (说 明 : 即 服务 器 端 收 到 客户 端 请 求 ) 的 时 间 ， 使 用 $_ 


| SERVER[REQUEST TIME] 要 优 于 time()。 


(10) 在 项 目 中 尽量 应 用 PHP 内 置 函 数 实 现 功能 ， 同 时 尽量 使 用 函数 代替 正则 表达 式 完 
成 相同 功能 。 

(11) str_ replace 函数 比 preg replace 函数 快 ， 但 strtr 函数 的 效率 比 str_ replace 函数 还 要 高 。 

(12) 使 用 选择 分 支 语 名 (switch) 优 于 使 用 多 个 过 ，else 站 语句 。 

(13) 不 提倡 使 用 @ 屏 蔽 错误 消息 ， 它 的 效率 非常 低 。 

(14) 开启 Apache 的 mod_deflate 模块 ， 可 以 提高 网 页 的 浏览 速度 。 

(15) 在 连接 数据 库 时 ， 尽 量 使 用 短 连接 ， 要 及 时 关闭 使 用 完毕 的 数据 库 连 接 。 

(16) 在 方法 中 递增 局 部 变量 ， 速 度 是 最 快 的 。 几 乎 与 在 函数 中 调用 局 部 变量 的 速度 相当 。 

(17) 派生 类 中 的 方法 运行 起 来 要 快 于 在 基 类 中 定义 的 同样 的 方法 。 如 果 能 将 类 的 方法 定 
义 成 static， 就 尽量 定义 成 static， 它 的 速度 会 提升 很 多 。 

(18) 尽量 少 进行 文件 操作 ， 虽 然 PHP 的 文件 操作 效率 也 不 低 。 

(19) Apache 解析 一 个 PHP 脚本 的 时 间 要 比 解 析 一 个 静态 HTML 页 面 慢 很 多 。 尽 量 多 用 
静态 HIML 页 面 ， 少 用 PHP 脚本 。 

(20) 除非 脚本 可 以 缓存 ， 否 则 每 次 调用 时 都 会 重新 编译 一 次 。 引 入 一 套 PHP 缓存 机 制 
通常 可 以 提升 25%~100% 的 性 能 ， 以 免除 编译 开销 。 
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(21) 当 操作 字符 串 并 需要 检验 其 长 度 是 否 满足 某 种 要 求 时 , 第 一 想法 是 使 用 stlen0 函 数 。 | 
此 函数 执行 起 来 速度 快 ， 因 为 它 不 做 任何 计算 ， 只 返回 在 zval 结构 〈C 的 内 置 数据 结构 ， 用 于 | 
存储 PHP 变量 ) 中 存储 的 已 知 字符 串 长 度 。 但 是 ， 由 于 strlen() 是 函数 ， 多 多 少 少 会 有 些 慢 ， 
因为 函数 调用 会 经 过 诸多 步骤 。 在 某 些 情况 下 ， 可 以 使 用 isset0 加 速 执行 代码 。 例 如 : 
<?php 
if(strlen($fo0) < $) { echo "Foo is too short" $$ } 
if(lisset($foo{5})) { echo "Foo is too short" $$ } 
?> 
调用 isset0 惟 巧 比 stlen0 快 ， 因 为 与 stlen0 不 同 的 是 ，issetO 作 为 一 种 语言 结构 ， 意 味 着 | 
它 的 执行 不 需要 函数 查找 和 字母 小 写 化 。 也 就 是 说 ， 实 际 上 在 检验 字符 串 长 度 的 顶层 代码 中 没 | 
有 人 花 太 多 开销 。 | 
(22) 当 执行 变量 $i 的 递增 或 递减 时 ，S$i++ 会 比 ++$i 慢 一 些 〈 仅 限 PHP 语言 )。 | 
++$i 更 快 是 因为 它 只 需要 3 条 指令 (opcodes)， 而 $it+ 需 要 4 条 指令 。 后 置 递增 实际 上 会 | 
产生 一 个 临时 变量 ， 这 个 临时 变量 随后 被 递增 ， 而 前 置 递增 直接 在 原 值 上 递增 。 | 
(23) 并 不 是 事 必 面向 对 象 (OOP)， 面 向 对 象 往往 开销 很 大 ， 每 个 方法 和 对 象 调用 都 会 | 
消耗 很 多 内 存 。 | 
(24) 如 果 可 以 通过 file_get_contents 替代 fle、fopen、feof、fgets 等 系列 方法 的 情况 下 ， | 
尽量 用 file_get_contents， 因 为 它 的 效率 要 高 很 多 。 但 是 ， 要 注意 file_get_contents 在 打开 一 个 | 
URL 文件 时 的 PHP 版 本 问题 。 | 


n 


问题 32 ”为 什么 页 面 之 间 无 法 传递 变量 ? 


问题 阐述 
为 什么 在 表单 中 提交 的 数据 ， 在 其 他 的 页 面 中 无 法 获取 ? 
专家 解答 


getpostsession 在 最 新 的 PHP 版 本 中 自动 全 局 变量 是 关闭 的 (在 phpini 中 ,register_globals | 
的 值 为 Off)， 如 果 要 从 上 一 页 面 获取 提交 的 变量 ， 则 必须 使 用 $_GET[]、$ PosT[" 或 
$_SESSION["]。 | 


专家 点 评 
从 PHP4.2.0 开始 register globals 的 默认 值 为 OE， 这 样 一 来 ， 以 前 的 很 多 可 以 直接 使 用 的 | 
变量 ， 如 SPHP_SELF 或 者 设 定 的 SESSION 变量 都 不 能 用 “$ 变 量 名 ”的 形式 访问 了 ， 这 可 能 会 | 


给 用 户 带 来 很 多 不 便 ， 但 却 有 助 于 安全 性 的 提高 。 访 问 这 些 变量 ， 就 需要 使 用 PHP 全 局 变量 。 
1.$_POST[] 全 局 变量 
使 用 PHP 的 $ POST 预定 义 变量 可 以 获取 表单 元 素 的 值 ， 格 式 为 : 

$_ POST[mame'] 


Ds L_ 
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2. $_GET[] 全 局 变量 
使 用 PHP 的 $_ GET[] 预 定义 变量 可 以 获取 通过 GET 方法 传 过 来 的 值 ， 格 式 为 : 
$_ GET[mame'] 
3. $_SESSION[] 变 量 
使 用 $_SESSION[] 变 量 可 以 获取 表单 元 素 的 值 ， 格 式 为 : 
$_SESSION['name'] 
使 用 $_SESSION[] 传 参 的 方法 获取 的 变量 值 ， 保 存 之 后 任何 页 面 都 可 以 使 用 ， 但 这 种 方法 
很 耗费 系统 资源 ， 建 议 读者 慎重 使 用 。 
注意 : 
PHP 可 以 应 用 $ POST[] 或 $ GET[] 全 局 变量 来 获取 表单 元 素 的 值 。 但 值得 注意 的 是 ， 获 取 
的 表单 元 素 名 称 区 别 字母 大 小 写 。 如 果 读 者 在 编写 Web 程序 时 疏忽 了 字母 大 小 写 ， 那么 在 程序 
运行 时 将 获取 不 到 表单 元 素 的 值 或 弹出 错误 提示 信息 。 


问题 33 ”为 什么 Win32 下 Apache2 用 get 方法 传递 
中 文 参数 会 出 错 ? 


问题 讲述 

Win32 下 Apache2 用 get 方法 传递 中 文 参数 会 出 错 ， 如 testphp?a= 你 好 &b= 你 也 好 ， 传 弟 
参数 导致 一 个 内 部 错误 。 
专家 解答 

要 想 解 决 这 个 问题 ， 可 以 使 用 urlencode0 函 数 进行 转换 ， 如 "test.php?a=".urlencode( 你 
好 )."&b=".urlencode( 你 也 好 ) 
专家 点 评 


urlencode() 函 数 主要 是 用 于 返回 字符 串 类 型 , 该 函数 会 将 传 入 的 字符 串 进行 编码 urlencode() 
函数 与 rawencode() 函 数 不 同 的 是 ，urlencode() 函 数 会 将 “空格 ”转换 成 “+”;， 而 rawencode() 
函数 则 会 将 “空格 ”转换 成 “%20”。urlencode() 函 数 的 其 他 字符 串 转 换 方式 与 rawencode() 函 数 
的 字符 串 转换 方式 相同 。 
urlencode() 函 数 的 语法 如 下 : 
string urlencode (string str) 


注意 : 

在 使 用 get 方法 传递 网 址 时 ， 如 果 不 使 用 编码 方式 则 可 能 会 出 现 获取 的 数据 信息 不 完整 的 
问题 ， 因 此 ， 使 用 urlencode(O) 函 数 进行 编码 就 会 避免 该 问题 的 发 生 ， 而 且 又 能 保证 传递 数据 的 
安全 性 ， 可 谓 是 一 举 两 得 。 


问题 阐述 | 
在 处 理 提交 的 内 容 时 ， 经 常 需要 考虑 其 中 的 空白 内 容 。 如 果 提 交 数 据 中 存在 空白 ， 则 会 给 | 
应 用 程序 造成 问题 。 例 如 ， 错 误 地 包含 空白 内 容 会 导致 存储 的 用 户 名 或 电子 邮件 地 址 不 正确 ， 


因此 ,加 


问题 34 ”如 何 从 文本 中 移 除 宝 白 内容 ? 


1 果 能 够 移 除 提交 的 表单 值 中 的 空白 将 会 非常 有 用 。 


专家 解答 


在 PHP 中 提供 了 trim0 函 数 去 除 字符 串 左 右 两 边 的 空格 和 特殊 字符 ;提供 了 ltrim0) 函 数 去 
除 字符 串 左边 的 空格 和 特殊 字符 ;提供 了 rtrim0 函 数 去 除 字符 串 中 右边 的 空格 和 特殊 字符 。 ”| 


1，trim() 函 数 
trim0) 函 数 用 于 去 除 字 符 串 开始 位 置 以 及 结束 位 置 的 空格 ， 并 返回 去 掉 空格 后 的 字符 串 。 
其 语法 如 下 : 


string trim(string str [,string charlist]); 


trim0 函 数 的 参数 str 是 要 操作 的 字符 串 对 象 , 参数 charlist 为 可 选 参 数 , 指定 需要 从 指定 的 | 


字符 串 
数 charli 


说 明 : 


ph 删除 哪些 字符 ， 如 果 不 设置 该 参数 ， 则 所 有 的 可 选 字 符 都 将 被 删除 。trim0 函 数 的 参 | 
st 的 可 选 值 如 表 2.1 所 示 。 | 
表 2.1 trim() 函 数 的 参数 charlist 的 可 选 值 
参 数 值 说 了 明 

‘0 NULL， 空 值 

At tab， 制 表 符 

un 换行 符 

\x0B 垂直 制 表 符 

Yr 回 车 符 

as 空格 


除了 以 上 默认 的 过 滤 字 符 列表 外 ， 也 可 以 在 charlist 参数 中 提供 要 过 滤 的 特殊 字符 。 


ltrim0O) 函 数 


ltrim0) 函 数 用 于 去 除 字符 串 左 边 的 空格 或 指定 字符 串 。 其 语法 如 下 : 


3 


string ltrim(string str [,string charlist]); 
rtrim0O) 函 数 


rtrim0 函 数 用 于 去 除 字符 串 右边 的 空格 。 其 语法 如 下 : 


String rtrim(string str [,string charlist]); 


专家 点 评 


在 使 用 ltrim0 函 数 和 rtrim0 函 数 时 ， 默 认 情况 下 是 只 去 除 左 边 或 右边 的 空格 ， 想 要 去 除 特 


L_ 


人 ip 六 预 知道 的 300 个 问题 


| 殊 字符 需要 在 函数 中 注 明 。 


问题 33 如何 获 取 上 传 文件 的 后 组 ? 


ae 问题 阐述 


| 文件 上 传 功 能 几乎 是 所 有 网 站 所 必 备 的 ， 然 而 该 功能 对 于 服务 器 来 说 具有 很 大 风险 ， 因 此 
| 应 该 对 文件 的 大 小 和 类 型 进行 限制 。 可 以 通过 获取 上 传 文件 的 后 绷 来 限制 上 传 文件 的 类 型 。 
| 专家 解答 
| 获取 上 传 文件 的 后 级 可 以 通过 strrev0) 函 数 对 上 传 文件 的 名 称 进行 反 转 输出 ， 并 通过 
| explode0 函 数 以 “.” 为 分 隔 符 ， 对 文件 名 称 进行 分 割 ， 然 后 再 次 应 用 strev0 函 数 对 数组 中 的 第 
| 一 个 元 素 值 进行 返回 输出 ， 即 获取 到 的 就 是 上 传 文件 的 后 缀 。 
| strrev0) 函 数 实现 PHP 中 字符 串 的 逆序 输出 。 其 语法 如 下 : 
string strrev ( string string ); 

参数 string 指定 需要 反 转 的 字符 串 。 
| 例如 ， 创 建 PHP 脚本 文件 ， 当 单 击 “ 上 传 ”按钮 时 ， 首 先 ， 利 用 post 方法 接收 文本 框 中 
| 的 信息 。 其 次 ,利用 反 转 函数 将 文本 框 信息 数据 进行 反 转 ， 并 利用 字符 串 拆 分 函数 以 “.” 进 行 
| 分割。 然后 ， 将 分 割 结果 保存 在 数组 之 中 。 最 后 ， 将 指定 数据 再 次 反 转 取得 并 输出 上 传 文件 的 
| 后 级 名 称 。 其 代码 如 下 : 


<2php 
这 $_POST['sub]){ // 获 取 表 单 提交 的 数据 
$a = strrev($_ POST['text]); /字符 串 反 转 
$b = explode(".",$a): // 字 符 串 分 隔 
S$c = strrev($b[0]); 1/ 获取 文件 后 级 
echo "文件 后 级 为 :.$c"; 
} 
! Te 
| 专家 点 评 


利用 strev0 函 数 检索 字符 串 相对 较 麻 烦 ， 可 以 使 用 正则 表达 式 函 数 preg match0 来 完成 该 功能 。 
问题 36 ”如 何 实 现货 币 数据 的 格式 化 输出 ? 


”问题 阐述 
| 在 开发 电子 网 站 时 ， 在 前 台 页 面 上 输出 货币 数据 显示 的 是 整数 数字 ， 不 能 以 货币 的 格式 输 
| 出 ， 那 么 如 何 才 能 让 它 以 货币 的 格式 输出 呢 ? 
专家 解答 
金额 的 格式 化 输出 通过 函数 number formatO 实 现 。 其 语法 说 明 如 下 : 
| string number format ( float number , int decimals) 

4 *。56 。 


string number format ( float number, int decimals, string dec_point, string thousands_sep) 


mumber formatO 函 数 返 回 参数 number 格式 化 后 的 字符 串 ， 该 函数 可 以 有 一 个 、 两 个 或 4 | 
个 参数 ， 但 不 能 有 3 个 参数 。 如 果 只 有 一 个 参数 number， 则 number 格式 化 后 会 合 去 小 数 点 后 | 
的 值 ， 且 每 一 千 就 会 以 逗号 (，) 来 隔 开 ;， 如 果 有 两 个 参数 ， 则 number 格式 化 后 会 到 小 数 点 第 | 
decimals 位 ， 且 每 一 二 就 会 以 逗号 来 隔 开 ， 如 果 有 4 个 参数 ， 则 number 格式 化 后 会 到 小 数 点 ， 
第 decimals 位 ，dec_ point 用 来 替代 小 数 点 (.)，thousands sep 用 来 替代 每 一 千 隔 开 的 逗号 (，)。 


问题 37 如何 解决 用 substr0 函 数 对 中 文字 符 囊 
截取 时 的 乱码 ? 


问题 前 述 


使 用 substr0 函 数 是 按 字 节 截 取 字 符 串 的 , 在 截取 中 文字 符 串 时 由 于 一 个 汉字 由 两 个 字符 组 | 
成 ， 所 以 如 果 只 截取 一 个 字符 就 会 出 现 乱 码 。 | 


专家 解答 


通过 substr0 函 数 对 中 文字 符 串 进行 截取 会 出 现 乱码 的 问题 ， 那 么 我 们 可 以 在 substr0 函 数 
的 基础 上 封装 一 个 自 定义 函数 ， 在 进行 截取 之 前 判断 其 是 中 文字 符 串 还 是 英文 字符 串 ， 如 果 是 | 
中 文字 符 串 则 每 次 截取 两 个 字 节 ， 如 果 是 英文 字符 串 则 每 次 截取 一 个 字 节 ， 这 样 就 不 会 出 现 乱 | 
人 码 的 问题 了 。 自 定义 函数 的 语法 如 下 : | 
function msubstr($str, $start, $len) { //$str 指 的 是 字符 串 ，S$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 | 
的 是 长 度 | 
S$strlen = $start + $len: // 用 $strlen 存储 字符 串 的 总 长 度 〔 从 字符 串 的 起 始 位 置 到 字符 | 
串 的 总 长 度 ) | 
for($i= 0; $i< $strlen; $i++) { /通过 for 循环 语句 ， 循 环 读 取 字符 串 
让 (ord (substr ($str, $i, 1) ) > 0xa0) {// 如 果 字 符 串 中 首 个 字 节 的 ASCI 序数 值 大 于 0xa0， 
则 表示 为 汉字 | 
$tmpstr 二 substr ( $str, $i, 2 );/ 每 次 取出 两 位 字符 赋 给 变量 $tmpstr, 即 等 于 一 个 汉字 | 
$i++; /变量 自 加 1 | 


} else{ /如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 Stmpstr 
S$tmpstr .= substr ( $str, $i, 1 ); 
bl 
} 
Tetum $tmpstr: // 输 出 字符 串 


} 


参数 $str 是 指定 被 截取 的 字符 串 ; 参数 $strart 是 截取 的 开始 位 置 ， 参数 $len 是 截取 的 长 度 。 
返回 值 Stmpstr 是 截取 后 的 字符 串 。 


专家 点 评 


对 中 文字 符 串 的 截取 虽说 是 通过 自 定义 函数 来 完成 的 ， 但 是 其 根本 还 是 应 用 substr 函数 ， 
只 是 在 进行 字符 串 截 取 时 ， 对 字符 串 的 类 型 进行 了 判断 。 | 
对 截取 字符 串 中 首 个 字 节 的 ASCII 序数 值 进行 判断 ， 如 果 ASCII 序数 值 大 于 0xa0， 则 表 | 


yy L 


有 yp 六 预知 道 的 500 个 问 是 
示 为 汉字 ， 那 么 在 应 用 substr 函数 进行 截取 时 ， 就 以 两 个 字 节 为 单位 进行 截取 ; 如 果 ASCI 序 
数值 小 于 0xa0， 则 表示 为 英文 字符 串 ， 那 么 在 应 用 substr 函数 进行 截取 时 ， 就 以 一 个 字 节 为 单 
位 进行 截取 。 这 样 将 中 文字 符 串 和 英文 字符 串 分 隔 进行 截取 就 避免 了 出 现 乱 码 的 问题 。 


问题 38 ”如 何 获取 多 个 数组 元 素 的 键 名 ? 


| 问题 阐述 


专家 解答 


如 何 获取 多 个 数组 元 素 的 键 名 ? 


在 使 用 array_search(O) 函 数 查询 数组 中 指定 元 素 时 ， 如 果 查 询 的 元 素 在 数组 中 不 止 出 现 一 


| 次 ， 那 么 amay _search0 函 数 将 返回 第 一 个 匹配 的 键 名 。 


| 被 返回 。 


如 果 想 要 返回 所 有 匹配 的 键 名 , 则 可 以 使 用 aray_keys0 函 数 , 并 设置 可 选 参数 search_value。 
array_keys() 函 数 返回 数组 中 所 有 的 键 名 。 语 法 如 下 : 
array array_keys ( array input [, mixed search_value [, bool strict]] ) 
array_keys0) 返 回 input 数组 中 的 数字 或 者 字符 串 的 键 名 。 
如 果 指 定 可 选 参数 search_value， 则 只 返回 该 值 的 键 名 ; 否则 input 数组 中 的 所 有 键 名 都 会 


| 专家 点 评 


|] PHP 5 起 ， 可 以 用 strict 参数 来 进行 全 等 比较 (一)。 


问题 39 为 什么 $array[str] 错 了 ? 


问题 请 述 


如 果 使 用 标识 符 [] 定 义 数 组 时 ， 那 么 应 在 用 字符 串 表 示 的 数组 索引 上 加 引号 。 例 如 ， 用 


| $array['str] 而 不 是 $array[str]。 


专家 解答 


但 是 为 什么 $array[str] 错 了 呢 ? 例如 下 面 的 一 段 代码 : 


<2?php 

$array[str] = 'PHP 编程 宝典 '; // 定 义 数组 元 素 
echo $array[str]; // 输 出 元 素 值 
?> 


这 样 编写 的 代码 是 错 的 , 但 可 以 正常 运行 。 那 么 为 什么 错 了 呢 ? 原因 是 此 代码 中 有 一 个 未 


定义 的 常量 (str) 而 不 是 字符 串 ('str'， 注 意 引 号 )。 如 果 PHP 在 以 后 定义 此 常量 ， 而 不 幸 的 是 
| 你 的 代码 中 有 同样 的 名 字 。 它 能 够 运行 ， 是 因为 PHP 自动 将 裸 字 符 串 〈 没 有 引号 的 字符 串 且 


不 对 应 于 任何 已 知 符号 ) 转换 成 一 个 其 值 为 该 裸 字符 串 的 正常 字符 串 。 例 如 ， 如 果 没 有 常量 定 
义 为 str， 则 PHP 将 把 它 蔡 代 为 str 并 使 用 。 
。58 。 


这 并 不 意味 着 总 需要 为 键 名 加 上 引号 。 如 果 键 名 是 常量 或 变量 ， 则 不 需要 加 引号 ， 和 否则 会 | 


使 PHP 不 能 解析 它们 。 


问题 40 ”如 何 设置 PHP 的 错误 级 别 ? 


问题 阐述 


如 何 配 置 PHP 的 错误 级 别 ? 


专家 解答 
PHP 错误 级 别 的 设置 有 两 种 方法 : 

-种 是 直接 操作 php.ini 配置 文件 ， 定 位 到 “error reporting =” 配 置 选项 ， 直 接 设 置 它 的 级 别 。 

另 一 种 是 在 程序 中 通过 error_reporting() 函 数 设 置 错误 级 别 ,error_reporting() 函 数 的 语法 如 下 : 


int error reporting ( [int level] ) 


参数 level 指定 报错 的 级 别 。 如 果 参 数 level 未 指定 ， 则 当前 报错 级 别 将 被 返回 。 表 2.2 给 
出 的 是 参数 level 的 可 选 值 。 


表 2.2 参数 level 的 可 选 值 


值 常 量 说 阴 
1 E ERROR 报告 致命 性 的 运行 错误 
多 E WARNING 报告 运行 时 警告 〈 非 致命 性 错误 ) 
4 E PARSE 报告 编译 时 解析 错误 
i 报告 运行 时 提要 (此 类 错误 经 常 是 由 于 代码 中 的 bug 引 起 的 ， 也 

可 能 是 有 意 的 行为 造成 的 ， 如 未 初始 化 变量 ) 
16 |E CORE ERROR PHP 启 动 时 初始 化 过 程 中 的 致命 错误 
32 “|E CORE WARNING PHP 启 动 时 初始 化 过 程 中 的 警告 〈 非 致命 性 错误 ) 
64 |E COMPILE ERROR 编译 时 致命 性 错误 
128 ”|E COMPILE WARNING ”| 编译 时 警告 〈 非 致命 性 错误 
256 |E USER ERROR 用 户 自 定义 的 错误 信息 
512 |E USER WARNING 用 户 自 定义 的 警告 信息 
1024 |E USER NOTICE 用 户 自 定义 的 提醒 信息 
gp | ee 编码 标准 化 警告 ， 允 许 PHP 建 议 如 何 修改 代码 以 确保 最 佳 的 互 操 
作 性 和 向 前 兼容 性 ， 即 可 移植 性 

4096 |E RECOVERABLE ERROR | 与 E ERROR 类似 ， 报 告 所 有 致命 性 的 运行 错误 
8191 |E _ ALL 报告 所 有 的 错误 和 警告 〈 不 包括 E_ STRICT) 

例如 ， 下 面 的 代码 关闭 了 用 户 自 定义 的 错误 和 和 警告， 执行 了 某 些 操作 ， 然 后 恢复 到 原始 的 

报错 级 别 。 代 码 如 下 : 
<?php 
/禁用 错误 报告 


。59 。 


人 Pp 六 须知 道 的 300 个 问题 


CD 
error_reporting(0); 
/报告 运行 时 错误 
! error_reporting(E ERROR |E WARNING |E PARSE): 
// 报 告 所 有 错误 


且 几 a 


任意 数目 的 以 上 选项 都 可 以 用 “或 ”来 连接 (用 OR 或 | )， 这 样 可 以 报告 所 有 需要 的 各 


级 别 错误 。 
问题 41 PHP 有 哪 几 种 注释 形式 ? 
问题 阐述 


”程序 中 的 注释 要 用 简单 、 明 了 、 精 确 的 语言 表达 出 程序 的 处 理 说 明 。 避 免 每 行程 序 都 使 用 
| 注释 ， 可 以 在 一 段 程序 的 前 面 加 一 段 注释 ， 具 有 明确 的 处 理 逻 辑 ， 那 么 ， PHP 中 都 有 哪儿 种 注 
| 释 形 式 呢 ? 
| 专家 解答 
| 1. 文件 注释 

每 个 包含 PHP 代码 的 文件 必须 至 少 在 文件 顶部 包含 这 些 phpDocumentor 标签 ; 


/于 六 


* 文件 的 简短 描述 

来 

* 文件 的 详细 描述 如果 有 的 话 ).… … 
来 


*# LICENSE: 一 些 license 信息 
来 
* (@Ocopyright 2005 Zend Technologies 
* (license http://www.zend.com/license/3 0.txt PHP License 3.0 
* (Oversion $Id:$ 
| * @link http://dev.zend.com/package/PackageName 
| * @since File available since Release 1.2.0 
| #/ 


| 2. 类 注释 
| 每 个 类 必须 至 少 包含 这 些 phpDocumentor 标签 : 
| /相让 

* 类 的 简 述 

* 类 的 详细 描述 (如 果 有 的 话 ).….… 


* (Qcopyright 2005 Zend Technologies 
* (@license http://www.zend.com/license/3 0.txt PHP License 3.0 
* (Dversion Release: package_ version@ 


。60 。 


* @link http://dev.zend.com/package/PackageName 
* @since Class available since Release 1.2.0 
* (@deprecated Class deprecated in Release 2.0.0 
3 函数 注释 
每 个 函数 ， 包 括 对 象 方 法 ， 都 必须 有 最 少 包含 下 列 内 容 的 文档 块 (docblock): 
函数 的 描述 
所 有 参数 
所 有 可 能 的 返回 值 
因为 访问 级 已 经 通过 "public"、"private" 或 "protected" 声 明 ， 不 需要 使 用 "@access"。 
如 果 函 数 /方法 抛 出 一 个 异常 ， 则 使 用 @throws: 
@throws exceptionclass [description] 
4. 块 注释 | 
块 注释 通常 用 于 提供 对 文件 、 方 法 、 数 据 结构 和 算法 的 描述 ， 它 被 置 于 每 个 文件 的 开始 处 | 
以 及 每 个 方法 之 前 。 另 外 ， 块 注释 也 可 以 被 用 于 其 他 地 方 ， 如 方法 内 部 。 在 功能 和 方法 内 部 的 | 
块 注释 应 该 和 它们 所 描述 的 代码 具有 一 样 的 缩 进 格式 。 | 
块 注释 之 首 应 该 有 一 个 空 行 ， 用 于 把 块 注释 和 代码 分 割 开 来 ， 例 如 ; 
/* 
* 这 里 是 块 注释 
2 
5， 单行 注 释 | 
短 注 释 可 以 显示 在 一 行内 ， 并 与 其 后 的 代码 具有 相同 的 缩 进 层级 。 如 果 一 个 注释 不 能 在 一 | 
行内 写 完 ， 就 应 该 采用 块 注释 。 单 行 注 释 之 前 应 该 有 一 个 空 行 。 单 行 注释 的 例子 : | 
if (condition) { 
上 # 以 下 代码 运行 的 条 件 */ 


} 
6 尾 端 注释 | 
极 短 的 注释 可 以 与 它们 所 要 描述 的 代码 位 于 同一 行 , 但 是 应 该 有 足够 的 空白 来 分 开 代 码 和 | 
注释 。 若 有 多 个 短 注释 出 现 于 大 段 代码 中 ， 它 们 应 具有 相同 的 缩 进 。 尾 端 注 释 的 例子 : 


让 (ga 一 2){ 
return TRUE: /* 对 单一 条 件 的 说 明 */ 
} Else { 
retur isPrime($a); /* 其 余 的 条 件 */ 
} 
7. 行 末 注 释 
注释 界定 符 “//” 可 以 注释 掉 整 行 或 一 行 中 的 一 部 分 。 例 如 : 
if($Euspice> 1){ 
// 注 释 用 法 
} 
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用 户 名 和 密码 不 允许 为 空 等 。 除 了 在 客户 端 通过 JavaScript 脚本 控制 文本 框 不 为 空 之 外 ， 是 否 


人 Pp 六 须知 道 的 300 个 问题 


专家 点 评 
注释 虽然 必 不 可 少 ， 但 也 不 应 过 多 ， 不 要 被 动 地 为 写 注释 而 写 注释 。 


问题 42 ”如 何 使 用 empty0 函 数 刊 疡 文本 框 是 否 为 室 ? 


| 问题 阐述 


在 网 页 程序 的 很 多 数据 提交 页 面 中 , 都 不 允许 文本 框 内 容 为 空 , 例如 , 在 用 户 登 录 页 面 中 ， 


| 可 以 在 服务 器 端 进行 判断 呢 ? 
| 专家 解答 


1. 解决 思路 
在 服务 器 端 判断 表单 提交 的 值 是 否 为 空 ， 可 以 用 empty0 函 数 来 完成 。empty0 函 数 检查 变 


| 量 是 否 为 室 ， 如 果 变量 是 空 或 零 ， 则 返回 true。 当 变量 为 : "、0、"0"、null、false、array()、 


var $var 以 及 没有 任何 属性 的 对 象 时 都 将 被 认为 是 空 的 。empty0 函 数 的 语法 如 下 : 
bool empty ( mixed var) 
参数 var 为 输入 的 变量 。 
2. 应 用 示例 


应 用 empty0 函 数 验 证 会 员 登 录 页 面 中 的 用 户 名 和 密码 是 否 为 空 ， 如 图 2.1 所 示 。 当 “用 户 
名 ”和 “密码 ”文本 框 为 空 时 ， 单 击 “ 登 录 ” 按 钮 ， 对 提交 的 表单 信息 进行 验证 ， 运 行 结果 如 
图 2.2 所 示 。 


密 到 i 
EE 
图 2.1 用 户 登 录 页 面 图 2.2 文本 框 为 空 时 显示 的 信息 
本 示例 的 关键 代码 如 下 : 


<2php 
$user =$_ POST[user]: 
$password =$_ POST[password']: 
这 empty($user) | empty($password)){ 
echo "用 户 名 或 密码 不 能 为 空 <br /> <a hre 信 login.html> 返 回 </a>"; 
jelse{ 
echo "输入 的 用 户 名 为 : $user 密码 为 : $password"; 


} 
2> 
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问题 43 如何 使 用 strlen0 函 数 计算 文本 长 度 ? 


问题 阐述 

在 对 超 长 文本 进行 显示 时 ， 由 于 页 面 空间 有 限 ， 所 以 通常 对 其 进行 分 页 显示 ， 在 分 页 前 ， 
首先 需要 计算 文本 的 长 度 ， 然 后 再 对 文本 进行 分 页 处 理 。 
专家 解答 

1. 解决 思路 

获取 指定 字符 串 的 长 度 使 用 strlen0 函 数 ， 其 语法 格式 如 下 : 

int strlen(string str) 

参数 str 指定 统计 的 字符 串 。 

2. 应 用 示例 

实现 超 长 文本 的 分 页 显示 ， 其 中 通过 strlen() 
函数 获取 文本 的 长 度 ， 同 时 还 应 用 了 文件 系统 函数 
库 中 的 fopen0 函 数 、fgets0 函 数 和 字符 串 处 理 函数 
库 中 的 substr0 函 数 ， 以 及 数学 函数 库 中 的 ceil0 函 


数 。 其 运行 结果 如 图 2.3 所 示 。 图 2.3 应 用 strlen0 函 数 实现 超 长 文本 
关键 代码 如 下 : 的 分 页 显示 
<2php 


function msubstr($str, $start, $len){ 
$strlen=$start+ $len; 
for($1=0;$i<$strlen:$i++) { 
if(ord(substr($str,$i,1))>0xa0) { 
S$tmpstr.=substr($str,$1,2); 
$i++; 
jelse{ 
S$tmpstr.=substr($str, $1,1); } 


Teturn $tmpstr; 
} 


> 
读 取 文本 文件 中 的 数据 ， 应 用 strlen() 函 数 获取 数据 中 字符 串 的 长 度 进行 输出 ， 代 码 如 下 : 


<2php 

if($page){ 

$fp=fopen("file/file.txt","r"); 

$counter=fgets($fp); 
Slength=strlen($counter); 
$page_count=ceil($length/200): 
$fp=fopen("file/file.txt","r"); 
include("function.php"); 
$c=msubstr($counter,0,($page-1)*200); 
$cl=msubstr($counter,0,$page*200); 


0d L_ 


(本 > 六 预知 道 的 300 个 问题 


echo substr($c1,strlen($c),strlen($c1)-strlen($c)); 


) 
2> 


到 。 问题 御 如 何 使 用 copy0 酌 数 对 文件 进行 复制 ? 
问题 前 述 


| 文件 复制 是 将 某 一 文件 由 一 个 路 径 复制 到 另 一 个 路 径 下 , 通常 我 们 会 在 Windows 系统 下 完 
| 成 这 项 操作 。 那 么 如 何 使 用 copyO 函 数 来 实现 这 一 功能 呢 ? 
| 专家 解答 
| 1. 解决 思路 
| 对 文件 进行 复制 操作 可 以 使 用 copy0 函 数 ， 如 果 操 作成 功 返 回 ttme， 否 则 返回 false。 该 函 
| 数 的 语法 如 下 : 
| bool copy( string source, string dest) 
| 参数 source 为 必要 参数 ,指定 进行 复制 操作 的 文件 目录 及 文件 名 称 ; 参数 dest 也 为 必要 参 
| 数 ， 指 定 复制 后 文件 的 存储 目录 及 名 称 。 
| 2. 应 用 示例 
应 用 copy0 函 数 实现 将 用 户 所 选择 的 文件 复制 到 其 他 位 置 。 其 关键 代码 如 下 : 


<2php 
if($_POST["submit"]!=""){ 
$filename=$ POST["filename"]; 
$objfilename=$ POST["objfilename"]: 
if(@copy($filename,$obijfilename)){ // 对 文件 进行 复制 操作 
echo "<script>alert( 文 件 复制 成 功 !"):</script>"; 
}else{ 
echo "<script>alert(' 文 件 复制 失败 1"):</script>"; 
} 
} 


?> 


问题 45 ”如 何在 浏览 器 中 直接 输出 图 像 7 (fgetcO 函 数 ) 


问题 阐述 

| 当选 择 一 个 图 片 文件 后 ， 如 何在 网 页 中 直接 查看 图 像 呢 ? 

专家 解答 

1. 解决 思路 

”在 浏览 器 中 直接 输出 图 像 可 以 使 用 人 etc0 函 数 。 该 函数 主要 用 于 获取 文件 指针 处 的 字符 并 
| 返回， 如 果 当前 指针 在 文件 尾 则 返回 false。 语 法 如 下 : 
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string fgetc(resource handle) 
参数 handle 指 某 文 件 的 标识 指针 。 


2 应 用 示例 | 
应 用 他 ete0 函 数 可 以 将 图 像 直接 在 浏览 器 中 进行 显示 。 运 行 本 示例 ， 如 图 24 所 示 , 首先 。 国人、 


选择 要 显示 的 图 像 ， 然 后 单 击 “ 确 定 ”按钮 ， 即 可 在 浏览 器 中 显示 出 该 图 像 。 实 例 运行 结果 如 
图 25 所 示 。 
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图 2.4 应 用 fgetc0 函 数 输出 图 像 图 2.5 图 像 输出 结果 
关键 代码 如 下 : 
<?php 


$filename=$_POST["filename"]; 

Sextendname=strstr($filename,"."); | 

if(!($extendname—".gif"l|$extendname—".jpg"||$extendname—".bmp"||$extendname—".png"){ | 
echo "<script>alert(' 不 支持 这 种 格式 的 图 像 !):history.back():</script>"; | 
exit; // 判 断 是 否 为 常规 图 像 格式 


’ 
$fp=fopen($filename,"r+"); 
while (false ! 一 ($char = fegetc($fp)){ 
echo "$char"; 
) 


> 
问题 46 如何 关闭 危险 函数 ? 


问题 讲述 
如 果 打 开 了 安全 模式 ， 那 么 函数 禁止 是 可 以 不 需要 的 ， 但 是 为 了 安全 还 是 将 其 考虑 进去 。 | 
例如 ,不 希望 执行 包括 system0 等 能 够 执行 命令 的 php 函数 , 或 者 能 够 查看 php 信息 的 phpinfo0 | 
等 函数 。 那 么 该 如 何 禁止 它们 呢 ? | 
专家 解答 
disable_functions = system,passthru,exec,shell exec,popen,phpinfo 
如 果 要 禁止 任何 文件 和 目录 的 操作 ， 需 要 关闭 很 多 文件 操作 : | 
disable functions = chdir,chroot,dir,getcwd,opendir,readdir,scandir,fopen,unlink,delete,copy.mkdir, | 
Imdir rename,file,file_get_contents,fputs,fwrite,.chgrp,.chmod.,chown | 


以 上 只 是 列举 了 部 分 文件 处 理 函数 ， 也 可 以 把 上 面 执行 的 命令 函数 和 该 函数 结合 ， 就 能 够 | 
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抵制 大 部 分 的 phpshell 了 。 
如 果 将 上 述 这 两 部 分 综合 到 一 起 ， 可 以 很 好 地 控制 用 户 的 权限 ， 保 证 Web 项 目的 安全 。 


问题 47 ”如 何 将 数组 中 的 元 素 合 成 字符 串 ? 


问题 阐述 

如 何 将 数组 中 的 元 素 合成 字符 串 ? 
专家 解答 

既然 可 以 对 字符 串 进行 分 割 ， 返 回 数组 ， 那 么 就 一 定 可 以 对 数组 进行 合成 ， 返 回 一 个 字符 
串 。 这 就 是 implode0 函 数 ， 将 数组 中 的 元 素 组 合成 一 个 新 字符 串 。 语 法 如 下 : 

string implode(string glue, array pieces) 

参数 glue 是 字符 串 类 型 ， 指 定 分 隔 符 ; 参数 pieces 是 数组 类 型 ， 指 定 要 被 合并 的 数组 。 

例如 ， 应 用 implode0 函 数 将 数组 中 的 内 容 以 * 为 分 阳 符 进行 连接 ， 从 而 组 合成 一 个 新 的 字 
符 串 ， 代 码 如 下 : 


<2php 

$str="PHP 编程 宝典 *NET 编程 宝典 *ASP 编程 宝典 *JSP 编程 宝典 “ ”// 定 义 字 符 串 常量 
$str_arr=explode("*",$str); // 应 用 标识 * 分 割 字符 串 
$array=implode("*",$str_arr); // 将 数组 合成 字符 串 
echo $array; // 输 出 字符 串 

?> 


运行 结果 为 : PHP 编程 宝典 *NET 编程 宝典 *ASP 编程 宝典 *JSP 编程 宝典 
问题 48” 如何 解决 使 用 empty 检查 函数 返回 结果 时 报错 ? 


问题 前 述 
在 PHP 开发 时 ， 当 使 用 empty 检查 一 个 函数 返回 的 结果 时 会 报错 : Fatal error: Can't use 


function returm value in write context。 例 如 : 


<2php 
echo empty(strlen(be-evilLorg)): 
> 


专家 解答 
查看 PHP 手册 ， 在 empty 函数 描述 的 地 方 有 以 下 文字 : 
Note: empty() only checks variables as anything else will result in a parse error. In other words, the 
following will not work: empty(trim($name)). 
empty() 只 检测 变量 ， 检 测 任何 非 变量 的 东西 都 将 导致 解析 错误 ! 
因此 ， 我 们 不 能 用 empty0 函 数 来 直接 检测 函数 返回 的 值 ， 上 面 例子 的 解决 方案 如 下 : 
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<?php 


Slength = strlen('be-evil.org’); // 将 返回 值 定义 到 变量 中 
echo empty($length); // 检 测 变量 
?> 


问题 49 ”如 何 使 用 mb substr 函数 截取 中 文字 符 串 ? 


问题 前 述 


通过 mb_substr 函数 可 以 对 中 文字 符 串 进行 截取 ， 并 且 不 会 出 现 乱码 的 问题 。 但 是 应 用 该 
函数 还 有 一 些 前 期 的 工作 要 做 ， 那 么 具体 都 有 哪些 工作 呢 ? 


专家 解答 


(1) 首先 ， 确 保 Windows/system32 下 有 php_mbstring.dll 文件 ， 如 果 没有 ， 可 以 从 php | 
安装 目录 extensions 中 复制 一 个 到 Windows/system32 中 。 | 
(2) 在 Windows 目录 下 找到 php.ini 文件 , 在 该 文件 中 搜索 mbstring.dll, 找到 下 面 的 语句 : | 
;extension=php_mbstring.dll 
把 前 面 的 :号 去 掉 ， 这 样 mb_substr 函数 就 可 以 生效 了 。 
mb_substr() 函 数 实现 对 指定 的 字符 串 进行 截取 操作 ， 其 语法 如 下 : 
string mb_substr ( string str, int start [, int length [, string encoding]] ) 
参数 说 明 : 
@ str: 指定 被 截取 的 字符 串 。 
@ start: 指定 截取 的 开始 位 置 。 
@@ length: 指定 截取 的 长 度 。 
@ encoding: 指定 被 截取 字符 串 的 编码 格式 。 
例如 ， 应 用 mb_substr0) 函 数 对 指定 的 字符 串 进行 截取 。 


<2php 

$str= "这 样 一 来 我 的 字符 串 就 不 会 有 乱码 ^_A; 
echo mb_substr($strutf8): 

2 


运行 结果 为 : 这 样 一 来 我 的 字符 串 就 不 会 有 乱码 ^ ^ 


问题 50 如何 解决 PHP 无 法 执行 system() 函 数 的 问题 ? 


问题 讲述 
PHP 无 法 执行 systemO 函 数 是 什么 原因 呢 ? 
专家 解答 


PHP 无 法 执行 system0O 函 数 ， 可 能 有 以 下 几 种 情况 : 


Sy Na 
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(1) Waming: system() has been disabled for security reasons 
该 错误 是 由 于 服务 器 安全 配置 将 system 函数 关闭 了 ,打开 php.ini 文件 ,将 disable_functions 
等 号 后 面 的 system 去 掉 即 可 。 
(2) Waming: system() [function.system|: Unable to fork 
该 错误 是 由 于 当前 站 点 的 运行 账号 无 法 访问 cmd.exe， 找 到 Windows\system32\cmd.exe 属 
性 ， 添 加 当前 站 点 的 运行 账号 权限 或 users 组 权限 即 可 。 
(3) 在 Web 访问 下 执行 没有 任何 响应 也 不 报错 ， 使 用 php cli 命令 行 执行 却 可 以 正常 返回 
结果 ， 该 错误 一 般 是 由 Windows 系统 路 径 引 起 的 ， 程 序 中 可 能 是 使 用 了 绝对 路 径 去 访问 程序 ， 


| 例如， 


System(D:\serveNDLLWECaptexe --url=http://tech.cncems.com --out="D:\web\tt.fei.cn\htdocs\ttt\tech.png""); 


此 时 用 命令 行 执 行 结果 正确 ， 但 用 Web 访问 就 会 返回 空白 页 ， 没 有 错误 。 
修改 方法 : 将 文件 的 访问 路 径 改 为 相对 路 径 , 并 将 正 Captexe 复制 到 站 点 程序 当前 目录 下 ， 


| 并 将 代码 改 为 ; 


System(IECapt.exe --url=http://www.baidu.com --out=t/tech.png); 
再 次 访问 页 面 ， 就 可 以 正常 输出 了 。 


问题 S1 ”如何 对 查询 关键 字 进 行 描 红 ? 


问题 阐述 


很 多 网 站 都 具有 查询 功能 ， 使 用 该 功能 可 以 让 用 户 很 快 地 找到 自己 所 需要 的 内 容 ， 但 传统 


的 搜索 只 是 将 符合 条 件 的 记录 显示 出 来 ， 这 样 使 用 户 很 难 在 记录 中 找到 自己 需要 的 关键 技术 ， 
| 所 以 ， 为 了 满足 用 户 的 需求 ， 就 产生 了 关键 字 描 红 技 术 。 那 么 该 技术 是 如 何 实现 的 呢 ? 


| 专家 解答 


1. 解决 思路 
对 查询 关键 字 进 行 描述 可 以 使 用 str_replace() 函 数 来 替换 指定 字符 串 , 从 而 实现 关键 字 的 描 


| 红 技术 。str_replace0O 函 数 执行 字符 串 蔡 换 操作 。 其 语法 如 下 : 


mixed str_replace ( mixed search, mixed replace, mixed subject [, int &count] ) 


参数 说 明 : 

@ search: 必要 参数 ， 指 定 需要 查找 的 字符 串 。 

@ replace: 必要 参数 ， 指 定 替 换 的 值 。 

日 subject: 必要 参数 ， 指 定 的 查找 范围 。 

@ count: 可 选 参数 ， 获 取 执行 蔡 换 的 数量 。 

2. 应 用 示例 

本 例 实现 了 一 个 简单 的 搜索 功能 ,输入 要 搜索 的 关键 字 , 单 击 “ 搜 索 ” 按 钮 , 将 通过 POST 
| 为 接收 到 数据 . 首先 进行 描 红 处 理 , 其 次 应 用 str_replace0 函 数 对 指定 字符 串 信 息 的 指定 内 容 


| 进行 普 换 。 运行 结果 如 图 2.6 所 示 。 
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图 2.6 对 查询 关键 字 进 行 描述 


关键 代码 如 下 : 
<2php 
这 $_POST[sub]){ 
$a = "<b style='color:red;font-size:18px:'>".$_POST[text]."</b>"; 
echo str_replace("$ POST[text]",$a.$stD; 
} 
?> 
专家 点 评 


对 查询 关键 字 进 行 描 红 在 网 站 搜索 中 经 常用 到 ， 实 现 这 一 功能 并 不 复杂 ， 和 希望 读者 可 以 熟 
练 掌握 。 | 


问题 $2 ”如 何 对 字符 串 与 HTML 进行 转换 ? 


问题 阐述 
字符 串 与 HTML 之 间 的 相互 转换 是 非常 简单 的 ， 通 过 转换 可 以 极 大 地 方便 Web 浏览 器 的 
浏览 和 字符 串 的 输出 。 | 
专家 解答 
1， 解 决 思路 
该 技术 的 实现 主要 应 用 htmlentities() 和 htmlspecialchars( 〇 函数 。htmlentities0 函 数 将 所 有 的 | 


字符 串 都 转换 成 HTML 字符 串 ， 而 htmlspecialchars() 函 数 只 是 将 字符 串 中 的 某 些 特殊 字符 转换 | 
成 HTML 格式 。 | 


htmlentities() 函 数 的 语法 如 下 : 
string htmlentities(string string,[int quote_style].[string charset]) 
参数 说 明 如 表 2.3 所 示 。 
说 明 : 


在 应 用 htmlentities() 函 数 对 字符 事 到 HTML 的 转换 过 程 中 ， 设 置 参数 charset 的 字符 集 为 | 
“GB2312”， 可 以 避免 在 输出 中 文字 符 时 出 现 乱码 的 问题 。 其 前 提 是 页 面 是 gb2312 编码 ， 如 | 
果 页 面 是 utf 8 编码 格式 ， 那 么 这 个 值 就 要 设置 为 tf 8。 | 


*。69 。 


5 Pip 和 需 知 道 的 300 个 辣 是 
| te 


表 2.3 htmlentities() 函 数 的 参数 说 明 


参 数 说 明 
| strin 必要 参数 ， 指 定 要 转换 的 字符 串 
仿 直 | 可 选 参数 ， 选 择 如 何 处 理 字符 串 中 的 引号 ， 有 3 个 可 选 值 ，(1) NET_COMPAT,， 转换 双 引 
一 全 | quote style | 号 ， 忽 略 单 引号 ， 它 是 默认 值 ， (2) ENT NOQUOTES， 忽 略 双 引号 和 单 引 号 ， (3) ENT_ 
charset 


够 避免 转换 中 文字 符 出 现 乱码 的 问题 


htmlspecialchars0 〇 函数 的 语法 如 下 : 
string htmlspecialchars(string string) 
| 该 函数 将 一 些 特殊 字符 转换 成 HTML 格式 ， 而 不 会 将 所 有 字符 都 转换 成 HTML 格式 。 该 
| 函数 转换 的 特殊 字符 如 表 2.4 所 示 。 
表 2.4 htmlspecialchars() 函 数 转换 的 特殊 字符 


QUOTES， 转 换 双 引号 和 单 引 号 
| 可 选 参数 ， 确 定 转换 所 使 用 的 字符 集 ， 默 认 字符 集 是 “ISO-8859-1”， 指 定 字符 集 后 就 能 


参 数 说 了 明 
& (和 ) 转 成 &amp 
"w( 双 引号 ) 转 成 &quot 
> (大 于 号 ) 转 成 &et 
< (小 于 号 ) 转 成 &lt 


2. 应 用 示例 


下 面 应 用 np HTML 字符 串 ， 并 且 设 置 转换 的 字符 集 为 
“GB2312”， 避 免 中 文 输出 乱码 的 问题 。 运 行 结果 如 图 2.7 所 示 。 


Shi idl? 30r bid eatp id 1, coliche nl? 
borderca) pee olor= W0198FF'> 
结晶 为 .se i 人 -用 今 
过 的 结 时 为: 日 的 了 作 、 和 chd> <fk> cr> <td 
align="center" bgcolor 一 >cimg 
Src='magesfbegJP 了 GT>citd> <fr> <fable> 


明日 科技 -- 用 今日 的 辛勤 工作 ， 搁 绷 日 百 倍 回报 


省 有 入 a 


区 公告 Eee 外 理由 子 


| 图 2.7 将 字符 转换 为 HIML 
关键 代码 如 下 : 


<?php 
$str='<table width="300" border="1" cellpadding="]1" cellspacing="]1" bordercolor="#FFFFFF" 
| bgcolor="#0198FF"> 
<tr> 
<td align="center" height="35" bgcolor="#FFFFFF"> 明 日 科技 -- 用 今日 的 辛勤 工作 ， 换 
明日 百倍 回报 ! </td> 
</tr> 
| <tr> 
| <td align="center" bgcolor="#FFFFFF" ><img strc="images/beg.JPG"></td> 
| </tr> 


了 *。70 。 
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</table>'; 


echo htmlentities($str,ENT_QUOTES,"utf-8")."<br>"; ”// 设 置 转换 的 字符 集 为 "GBK" 
?> 
<?php echo $str; ?> 


专家 点 评 天 
字符 串 与 HTML 相互 转换 的 技术 应 用 最 广泛 的 地 方 就 是 输出 数据 库 中 的 数据 ， 例 如 ， 当 
数据 库 中 存储 的 是 一 个 可 以 运行 程序 的 代码 时 ， 如 果 没 有 对 该 数据 进行 转换 而 直接 输出 ， 那 么 


输出 的 结果 将 是 该 程序 的 运行 结果 ， 而 不 是 程序 代码 。 目 前 在 论坛 、 博 客 〈 帖 子 浏览 》， 特别 | 
与 编程 有 关 的 论坛 或 博客 中 应 用 最 为 广泛 。 | 


问题 53 ”如何 异 蔽 违禁 词语 ? 


问题 前 述 | 

很 多 论坛 、 评 论 、 贴 吧 都 有 过 滤 敏 感 词 的 功能 。 所 谓 的 “敏感 词 ”是 指 网 站 管理 者 为 了 正 | 
常 经 营 发 展 、 减 少 不 必 要 的 纷争 以 及 消除 不 良 影响 ， 而 人 为 设 定 的 一 些 较 易 引起 人 们 歧义 思维 | 
的 词汇 。 屏 项 违禁 词 主要 是 为 了 使 网 站 的 内 容 健康 而 规范 ， 这 项 功能 在 论坛 发 表 中 比较 常用 。 | 
如 果 不 设 置 违禁 词 屏蔽 ， 用 户 随 意 发 表 不 健康 的 内 容 ， 会 对 网 站 造成 很 大 的 不 良 影响 。 那 么 屏 | 
项 违禁 词语 的 功能 是 如 何 实现 的 呢 ? | 


专家 解答 
1， 解 决 思路 


实现 违禁 词 屏蔽 功能 的 方法 有 很 多 种 ,例如 ,可 以 应 用 fleO 函 数 读 取 存储 在 文本 文件 中 的 
敏感 词汇 ， 并 将 其 存储 在 $file_ word 数组 中 。 然 后 ， 应 用 for 循环 语句 自动 读 取 数组 元 素 (敏感 | 
词 )。 最 后 通过 正则 表达 式 preg_match0 函 数 检验 用 户 提交 的 留言 信息 中 是 否 含 有 敏感 词 。 | 


在 正 浏 览 器 中 输入 地 址 ， 按 Enter 键 ， 在 用 户 提交 留言 信息 后 ， 系 统 将 签 写 的 留言 信息 与 | 
本 模块 设 定 的 敏感 词 进行 一 一 对 比 ， 如 果 在 用 户 签 写 的 留言 信息 中 存在 敏感 词 ， 如 “喜欢 ”， 
就 会 弹出 留言 中 含有 敏感 词 的 警告 信息 ， 结 果 如 图 2.8 所 示 。 


司 
Ea ili 3 习 


图 2.8 过 滤 敏 感 词 “ 喜 欢 ” 
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Fr 
SC 
示例 代码 参考 如 下 : 
| <2php 


1. Language0 函 数 ， 只 接受 文本 框 和 文本 域 的 两 组 信息 。 
2. 本 函数 需 调用 自 定义 函数 con0 连 接 数据 库 。 

3. 将 违禁 词语 保存 在 数组 中 。 
4 


oa . 通过 正则 表达 式 函数 进行 筛选。 
| 2 


function language($strr, $strrl){ // 屏 蔽 标题 信息 
con("localhost","root","111","db_database","utf-8"); 
$array=array(" 喜 欢 "," 喜 爱 "," 笔 记 本 电脑 7); 
for($a=0:$a<count($array):$a+H){ 
if(preg_match("/$array[$al]/",$strr){ 
echo "<script>alert( 标 题 内 容 包含 违禁 词语 或 为 室 ， 请 更 改 );</script>"; 
break; 
}else{ 
if($strrl!=""){ 
| mysql_query("insert into tb_demo02(id,title,content)values(",'$_POSTI[text]', 
| '$_POST[tear])"): 
echo "<script>alert(' 发 布 成 功 ');location.href~'index.php?id=2'</script>"; 
break: 
}else{ 
echo "<script>alert(' 发 布 内 容 为 空 );</script>"; 
break; 
} 
b 
} 
1 
人 
自 定义 函数 connO 预 期 接收 5 个 参数 ， 其 中 包括 数据 源 、 数 据 库 用 户 名 和 密码 、 连 接 的 数据 库 
名 称 和 页 面 的 编码 格式 。 
5 
function con($local,$root,$pwd,$db, $code){ 
S$conn=mysql_connect($local,$root,$pwd) or die ("Connect MySQL False"); 
mysql_select_db($db,$conn) or die ("Connect DB False"): 
mysql_query($code); 


用 户 在 论坛 系统 中 发 表 主 题 时 ， 如 果 使 用 了 不 恰当 的 词语 ， 系 统 往往 会 提示 用 户 用 了 违禁 
， 的 词语 ， 因 此 发 表 没有 成 功 ， 请 更 改 等 ; 或 者 还 有 些 用 户 尽管 系统 提示 发 表 成 功 ， 但 是 在 浏览 
| 信息 时 发 现 发 表 的 信息 内 容 被 系统 自动 屏蔽 为 不 可 见 的 模式 了 。 


| 问题 54 如何 对 字符 串 进 行 分 着 及 乏 接 ? 


| 问题 阐述 
| 字符 串 的 连接 与 分 割 是 非常 重要 的 两 个 内 容 , 通过 其 可 以 将 数组 按照 指定 的 规则 转换 成 字 
】 。72 。 
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符 串 ， 也 可 以 将 字符 串 按 照 指 定 的 规则 进行 分 割 ， 返 回 一 个 数组 。 其 应 用 范围 非常 广 ， 例 如 网 
站 的 购物 车 或 在 线 投票 系统 等 。 那 么 这 两 个 功能 具体 是 如 何 实现 的 呢 ? | 
专家 解答 
1. 解决 思路 
对 字符 串 进 行 分 割 或 连接 应 用 的 是 implode0 和 explode0 〇 函数。 
(1) explode0 函 数 ， 按 照 指定 的 规则 对 一 个 字符 串 进行 分 割 ， 返 回 值 为 数组 。 语 法 如 下 : | 
array explode ( string separator, string string [, int limit]); | 
参数 string string 表示 分 割 条 件 ; 参数 separator 表示 指定 的 分 隔 符 ; 参数 int limit 表示 指定 | 
生成 数组 的 最 大 值 。 | 
(2) implode() 函 数 ， 与 explode 函数 配合 使 用 ， 将 数组 的 内 容 组 合成 一 个 字符 串 。 其 语法 | 


如 下 : 
string implode ( string glue, array pieces); 
参数 glue 是 字符 串 类 型 ， 指 定 分 隔 符 ; 参数 pieces 是 数组 类 型 ， 指 定 要 被 合并 的 数组 。 
2. 应 用 示例 
(1) 字符 串 的 分 割 | 
本 示例 开发 一 个 在 线 投票 管理 系统 ， 通过 向 文本 框 中 输入 要 进行 投票 的 内 容 ， 并 上 且 以 “**” | 
作为 每 项 内 容 之 间 的 分 割 符 。 单 击 “ 提 交 ” 按 钮 ， 将 在 投票 选项 中 的 内 容 进行 输出 ， 运行 结果 | 
如 图 2.9 所 示 。 | 
关键 代码 如 下 : 
<2php 
$strl=$_POST[strl]; 
$str2=$_POST[str2]:; 
$array=explode($str2,$str1); 
echo "该 字符 串 分 割 结果 为 :"."<br>"; 
while(list($name,$value)=each($array)){ 
echo $value."<br>"; 
} 
?> 
(2) 字符 串 的 连接 | 
implode(0) 函 数 将 数组 按照 指定 的 规则 转换 成 一 个 字符 串 ， 实 现 根据 数据 库 中 的 数据 表 动 态 | 
地 生成 数据 录入 的 页 面 , 其 中 主要 应 用 implode0 函 数 将 表单 中 提交 的 数据 存储 到 数据 库 中 。 运 | 
行 结果 如 图 2.10 所 示 。 | 


EDIELEE Pp 
按 以 下 字符 进行 : 一 人 全: 
有 限 公司 吉林 省 8 长 碍 市 8 二 道 区 
图 2.9 字符 串 的 分 割 图 2.10 字符 串 的 连接 
关键 代码 如 下 : 


人 Pip 六 预 知道 的 300 个 问题 


<?php 
$strl[0]-$ POST[strl1]: 
$strl[1]=$_ POST[strl2]:; 
| $strl[2]-$_ POST[strl3]:; 
| $str2=$_POST[str2]: 
鲜 | | $str=implode($str2,$str1); 
echo "该 字符 串 合并 结果 为 : "."<br>"; 


a 


?> 


问题 55 如 何 将 卫 地址 转换 成 对 应 的 数字 ? 


| 问题 阐述 
| IP 地 址 是 互联 网 身份 的 象征 , 通过 它 可 以 得 到 用 户 的 相关 信息 。 很 多 网 站 都 会 记录 访问 者 
| 的 了 瑟 地 址 ， 通 过 它 可 以 判断 用 户 的 所 在 地 ， 这 样 就 可 以 了 解 每 个 地 区 用 户 的 访问 量 了 。 
| 专家 解答 
| 1. 解决 思路 
| 通过 PHP 中 提供 的 SREMOTE ADDR 变量 可 以 获取 用 户 的 他 地址 ,但 并 不 能 判断 他 的 所 
| 在 地 , 为 了 解决 上 述 问题 可 以 先 将 该 他 转换 成 其 对 应 的 数字 , 之 后 根据 该 数值 查找 卫 所 在 地 。 
| 首先 需要 验证 输入 的 人 P 地 址 是 否 合 法 ， 如 果 合 法 ， 再 将 其 转换 为 对 应 的 数值 。 验 证 他 地 
| 址 是 否 合法 可 以 通过 正则 表达 式 实现 。 
| IP 地 址 使 用 4 位 地 址 定位 设备 ， 它 采用 点 分 十 进 制 数 的 格式 “xxx.xxx.xxXx.xXxXX”， 如 
| 192.168.1.66。 因 此 ， 验 证 瑟 地址 是 否 合法 的 正则 表达 式 如 下 : 
| /aD).0aD).0anD).0an)/ 

将 他 地 址 转换 为 对 应 的 数值 可 以 应 用 以 下 公式 实现 : 

Pl1*255*255*255+P2*255*255+P3*255+P4*1 

| 在 上 面 的 公式 中 ，P1 代表 了 P 地 址 第 1 段 的 数值 ，P2 代表 了 P 地 址 第 2 段 的 数值 ，P3 代表 
| 人 P 地 址 第 3 段 的 数值 ，P4 代表 也 地 址 第 4 段 的 数值 。 
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2. 应 用 示例 J es /1127 1 0 11or 7106/0 maw ID 
| 下面 通过 正则 表达 式 验 证 他 地 址 是 否 合 法 , 然后 将 mr 
| 了 P 地 址 转换 为 对 应 的 数字 ， 进 而 获取 卫 地 址 所 在 地 。 ED 旺 
| 运行 结果 如 图 2.11 所 示 。 

图 2.11 将 卫 地 址 转换 为 对 应 的 数字 

关键 代码 如 下 : . 

// 验 证 下 地 址 

<script language="javascript"> 

function deal(myform){ 
var ip=form!1 .ip.value; 
objExp=/dD.CdbCdD.CdDA 
这 ip 一 ""){ 


! alert(" 请 输入 卫 地 址 1");forml.ip.focusO:;return:; 
了 。74 。 


} 
if(objExp.test(ip){ 
myform.ipNum.value=convert(ip); 
jelse{ 
alert(" 您 输入 的 卫 地 址 不 合法 !):forml.ip.focusO:return: 
} 
} 


</script> 

// 将 四 地 址 转换 为 对 应 的 数字 

<script language="javascript"> 

function convert(ip){ 
ip=ip.split("."); 
vip=ip[0]*255*255*255+ip[1]*255*255+ip[2]*255+ip[3]*1: 
Teturn vip; 


b 
</script> 
专家 点 评 


将 卫 地 址 转换 为 对 应 的 数字 后 ， 如 果 要 查看 对 应 的 地 区 ， 可 以 到 网 上 查找 。 
问题 56 ”如 何 创建 新 型 字符 串 ? 


问题 阐述 | 
以 往 字符 串 的 定义 方式 都 是 将 字符 串 用 单 引号 或 双 引 号 括 起 来 进行 创建 的 ， 在 PHP 5.0 以 | 
后 的 版 本 中 出 现 了 一 种 新 型 字符 串 的 定义 方式 ， 下 面 进行 介绍 。 | 
专家 解答 
1. 解决 思路 
创建 新 型 字符 串 以 “<<<” 开 始 ， 后 面 紧 跟着 字符 串 开 始 标记 。 新 型 字符 串 格式 如 下 : 
<<< 标 记 
字符 串 内 容 /相当 于 "字符 串 内 容 " 
标记 ; 
2. 应 用 示例 | 
创建 PHP 5 新 型 字符 串 ， 本 例 以 strmark 作为 标记 为 例 ， 之 后 为 字符 串 的 内 容 ， 最 后 以 标 | 
记 加 分 号 “strmark:” 结 束 。 代 码 如 下 : | 


<?php 
$str=<<<strmark 
吉林 省 明日 科技 
strmark; 

echo $str; 

?> 


运行 本 示例 ， 如 果 在 浏览 器 中 显示 “吉林 省 明日 科技 ”， 则 说 明 程序 创建 成 功 。 


A L_ 
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问题 57 ”如何 对 字符 串 进行 加 窗 解 密 ? 


鲍 汪 问题 请 述 
对 于 电子 商务 网 站 的 用 户 来 说 ， 最 重要 的 就 是 账号 和 密码 的 安全 。 如 果 账 号 和 密码 未 经 过 
| 加 密 处 理 ， 则 大 大 增加 了 安全 隐患 ， 给 非法 用 户 以 可 乘 之 机 。 为 了 确保 用 户 的 利益 及 隐私 不 被 
| 泄露 ， 多 数 网 站 开始 对 用 户 的 账号 和 密码 进行 加 密 。 
专家 解答 
| 1. 解决 思路 
| 字符 串 的 base64 编码 与 解码 可 以 通过 PHP 的 预定 义 函 数 base64_encode() 和 base64_decode() 
| 来 实现 。 对 字符 的 base64 编码 用 函数 base64_encode0 实 现 ， 语 法 如 下 : 
| string base64_encode(string str) 
该 函数 实现 对 字符 串 str 的 base64 编码 。 
对 已 经 进行 base64 编码 的 字符 串 进行 解码 用 函数 base64_decode() 实 现 ， 语 法 如 下 : 
string base64_decode(string str) 
该 函数 实现 对 已 经 进行 base64 编码 的 字符 串 进 行 解密 。 
Re 
2. 应 用 示例 密码 ， | 
| 通过 字符 串 函 数 base64_encode()、base64_decode() EE3| 2 
| 实现 对 密码 的 加 密 和 人 解密， 运行 结果 如 图 2.12 所 示 。 


经 加 密 hak 


| 创建 PHP 脚本 ， 当 单 击 “ 提 交 ” 按钮 时 ， 通 过 字 POPI 
| 符 串 函数 base64_encode()、base64_decode() 实 现 对 字 ET 
da andi 图 2.12 用 户 注册 的 密码 进行 加 密 和 解密 
这 $_POST[sub]){ // 通 过 POST 方法 接收 参数 
echo "经 加 密 <hr>"; 
echo "用 户 名 : "$_POST[text]: 
S$str = base64_encode($ POST[pwd]): // 对 密码 进行 加 密 
echo "密码 : ". $str; 
echo "经 解密 "; 
echo "<hr>"; 
echo "用 户 名 : ".$_POST[text]: 
echo "密码 : ". base64_decode($str):; // 对 密码 进行 解密 
» 
| ?> 
| 专家 点 评 


| PHP 中 提供 了 很 多 加 密 和 解密 的 函数 ， 例 如 base64_encode0 或 md50 等 ， 在 实际 使 用 时 ， 
| 用 户 可 以 挑选 一 个 合适 的 函数 使 用 。 
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问题 58 ”如 何 比较 字符 串 的 大 小 ? 


问题 阐述 

在 Web 项 目 开发 过 程 中 ， 经 常 需要 确定 两 个 或 多 个 字符 串 之 间 的 大 小 关系 。 
专家 解答 

1. 解决 思路 


PHP 中 实现 字符 串 的 大 小 比较 可 以 通过 strcasecmp0) 函 数 和 stremp0) 函 数 来 实现 .语法 如 下 : 


int strcasecmp(string strl, string str2) 
int stremp(string strl, string str2) 


这 两 个 函数 用 于 比较 两 个 字符 串 的 大 小 (基于 ASCII 码 值 )， 如 果 strl>str2 则 返回 1， 如 | 
果 strl<str2 则 返回 -1; 如 果 strl=str2 则 返回 0。 二 者 的 区 别 在 于 函数 | | 


[xm 
| 


中 不 区 分 大 小 写 ， 而 函数 sttcemp0 在 比较 的 过 程 中 区 分 大 小 写 。 
2. 应 用 示例 


下 面 使 用 strcasecmp0) 函 数 和 stremp0 函 数 来 实现 字符 串 大 小 
的 比较 ， 运 行 结果 如 图 2.13 所 示 。 
关键 代码 如 下 : 


图 2.13 ”比较 字符 串 的 大 小 
<?php 
if($_POST["submit"]!=") 


$strl=$_POST[str1]; 
$str2=$_POST[str2]; 
Sresult=stremp($str1,$str2); 
if($result==1) 

echo "<div align=center> 字 符 串 ".$strl." > 字符 串 ".$str2."</div>"; 
if($result==0) 

echo "<div align=center> 字 符 串 ".$strl." = 字符 串 ".$str2."</div>"; 
if($result=—-1) 

echo "<div align=center> 字 符 串 ".$strl." < 字符 串 ".$str2."</div>"; 
} 


?> 
专家 点 评 | 
比较 字符 串 的 大 小 主要 集中 在 对 类 型 及 大 小 写 的 比较 上 ,在 实际 应 用 中 ， 当 遇 到 同一 个 英 | 


文 单词 ， 如 果 其 大 小 写 不 同 ， 会 使 运算 结果 出 错 ， 就 可 以 先 对 字符 串 进行 比较 ， 再 根据 情况 做 | 
下 一 步 的 操作 。 | 


问题 59 ”如 何 获取 上 传 文件 的 扩展 名 ? 


问题 阐述 


验证 上 传 文件 的 格式 类 型 是 对 上 传 文件 的 一 种 安全 保障 ,大 家 不 妨 设想 一 下 ,对 于 实现 任意 
Ry A SG 


(本 > 六 预知 道 的 300 个 问题 


| 格式 的 文件 上 传 ， 一 方面 会 给 非法 用 户 以 可 乘 之 机 ， 另 一 方面 也 给 服务 器 带 来 很 大 的 压力 。 可 
| 见 对 上 传 文件 格式 的 规范 是 很 有 必要 的 ， 实 现 这 一 功能 的 前 提 需 要 先 获取 上 传 文件 的 扩展 名 。 


专家 解答 
天 | ”| 加 火电 路 


获取 上 传 文件 的 扩展 名 可 以 应 用 ststr0 函 数 ， 从 上 传 文件 名 称 中 的 “.” 开 始 截 取 ， 即 可 获 
| 取 到 上 传 文件 的 后 缀 名 称 ， 然 后 与 程序 中 定义 的 可 上 传 文件 格式 进行 匹配 ， 如 果 匹 配 成 功 ， 则 
| 表示 该 文件 符合 上 传 条 件 。 
| strstr0 函 数 获取 一 个 指定 字符 串 在 另 一 个 字符 串 中 首次 出 现 的 位 置 到 后 者 未 尾 的 子 字符 
| 串 。 如 果 执 行 成 功 ， 则 返回 剩余 字符 串 〈 存 在 相 匹配 的 字符 )， 和 否则 返回 lse。 语 法 如 下 : 
| string strstr ( string haystack, string needle) 


参数 haystack 指定 从 哪个 字符 串 中 进行 搜索 ， 参数 needle 指定 搜索 的 对 象 。 如 果 该 参数 是 
-个 数值 ， 那 么 将 搜索 与 这 个 数值 的 ASCII 码 相 匹配 的 字符 。 


说明 ; 
| strstr0) 函 数 区 分 大 小 写 ， 如 果 不 需要 对 大 小 写 加 以 区 分 ， 可 以 使 用 stristr0) 子 数 。 


2.， 应 用 示例 
| 本 未 例 应 用 ststr0 函 数 获取 上 传 图 片 的 后 绥 ， 以 此 来 判断 上 传 图 片 的 格式 是 否 正确 ,如果 正 
| 确 则 将 图 书 上 传 到 服务 器 根 目录 下 的 upload 文件 夹 下 ， 耕 则 给 出 提示 信息 。 运 行 结果 如 图 2.14 
| 和 图 2.15 所 示 。 
各 Ewa 
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图 2.14 应 用 strstr0 检 索 上 传 图 片 的 后 绥 图 2.15 应 用 strstr0 检 索 上 传 图 片 的 后 绥 
其 关键 代码 如 下 : 
<form name="form" method="post" action= "index.php" enctype="multipart/form-data"> 
<input name="u file" type="file" size="24"/> 
(<span class="STYLE1">* 上 传 图片 是 .jpg 格式 ， 大 小 不 能 超过 1.2MB</span>) 
<input type="image" name="imageField" src="imges/sc.bmp" onClick="form.submit();"> 
</form> 
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<?php 

这 $_ FILES[u flel][name] 一 tmue){ 
$file path="./upload\": /定义 图 片 在 服务 器 中 的 存储 位 置 
S$picture name=$ FILES[u_file][name]:; // 获 取 上 传 图 片 的 名 称 


Spicture_name=strstr($picture_name , "."); // 通 过 strstr0 函 数 截 取 上 传 图 片 的 后 级 
if($picture_name!= "jpg" && $picture name!=".JPG" ){  // 根 据 后 级 判断 图 片 的 格式 
echo "<script>alert( 上 传 图 片 格式 不 正确 ,请 重新 上 传 windowlocationhref- ， 
"index.php';</script>"; 
}else if($ FILES[u filel[tmp_nameD){ 
move_uploaded file($ FILES[u file][tmp_name],$file path.$_FILES[u filel[name)); 


/执行 图 片上 传 | 
echo "<script>alert(' 图 片上 传 成 功 !"); window.location.href='index.php';</script>"; | 
}else{ | 
echo "<script>alert(' 上 传 图 片 失败 ! '"); window.location.href='index.php';</script>"; 
b 
} 
>. 


问题 60 如何 控制 页 面 中 输出 字符 串 的 长 度 ? 


问题 阐述 | 

在 网 页 程序 中 ， 经 常 需 要 显示 一 些 较 长 的 信息 ， 如 公告 信息 、 标 题名 称 等 。 如 果 对 这 些 信 
息 进 行 全 部 输出 ， 则 页 面 中 将 会 显示 出 长 短 不 一 的 字符 串 信息 ， 会 使 页 面 看 起 来 比较 杂乱 。 为 | 
了 保持 整个 页 面 布局 的 合理 、 美 观 ， 经 常 需要 对 这 些 较 长 的 字符 串 进行 部 分 输出 ， 以 保证 显示 | 
的 字符 串 长 度 一 致 。 对 于 超出 长 度 部 分 ， 可 以 使 用 指定 的 字符 进行 替换 。 | 
专家 解答 

1， 解 决 思路 | 

控制 页 面 中 输出 字符 串 的 长 度 可 以 使 用 str_replace0 函 数 和 substr0 函 数 。str replaceO 函 数 | 
主要 用 于 以 指定 的 字符 替换 超出 长 度 的 部 分 ，substr0) 函 数 主 要 实现 截取 字符 的 确定 位 置 。 

(1) str replace0 函 数 ， 实 现 字符 串 的 蔡 换 。 语 法 如 下 : 


mixed str_replace ( mixed search, mixed replace, mixed subject , int &count ) | 
str_replace() 函 数 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 蔡 换 , 参数 &count 表 | 

示 蔡 换 字 符 串 执行 的 次 数 。 | 
str_replace() 函 数 的 参数 说 明 如 表 2.5 所 示 。 | 


表 2.5 str_replace() 函 数 的 参数 说 明 | 


参 数 说 了 明 | 
search 指定 将 要 被 替换 的 字符 
replace 指定 替换 所 使 用 的 字符 | 
subject 指定 被 操作 的 字符 串 
&count 替换 字符 串 执 行 的 次 数 
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(2) substr0 函 数 ， 从 指定 的 字符 串 中 按照 指定 的 位 置 截取 一 定 长 度 的 字符 。 语 法 如 下 : 
string substr(string str,int start,int length) 
substr0) 函 数 的 参数 说 明 如 表 2.6 所 示 。 


表 2.6 substr() 函 数 的 参数 说 明 
CT 


说 明 
str 指定 字符 串 对 象 
start 指定 开始 截取 字符 串 的 位 置 。 如 果 参 数 start 为 负数 ， 则 从 字符 串 的 末尾 开始 截取 
length ”| 可 选 参 数 ， 指 定 截 取 字符 的 个 数 ， 如 果 参 数 length 为 负数 ， 则 表示 取 到 倒数 第 length 个 字符 
说 明 : 


本 轧 数 中 参数 start 的 指定 位 置 是 从 0 开始 计算 的 ， 即 字符 串 中 的 第 一 个 字符 表示 为 0。 

2， 应 用 示例 

本 例 实现 在 输出 最 新 动态 标题 时 截取 24 个 字 节 标题 名 称 , 超出 长 度 的 内 容 以 省 略 号 代替 ， 
运行 结果 如 图 2.16 所 示 。 
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图 2.16 控制 页 面 中 输出 字符 串 的 长 度 


| 关键 代码 如 下 : 
| <2php 
| include_once("conn/conn.php"); // 调 用 连接 数据 库 的 文件 
| include_once("function php"); 
| 
| <table width="365" height="22" border="0" align="center" cellpadding="0" cellspacing="0"> 
<?php 
$sql-mysql_ query("select * from tb_new_dynamic order by id desc limit 0.6",$id); 
while($myrow=mysql_fetch array($sql)){ 
> 
<tr> 


<td width="20" height="22"><div align="center"><img src="images/01.JPG"/></div></td> 
<td width="258" height="22"><a hre 人 ="new_dynamic php?id=<?php echo $myrow["id"]:?>" 


*。80 。 


title="<?php echo $myrow["dynamic title"]:?>"> 


<2php 
echo unhtml(msubstr($myrow["dynamic title"],0,24)); 
这 strlen($myrow["dynamic title"])>24){ 
Coho 0 
bh 
We 
</a> </td> 


<td width="87"><?php echo "<font color=red>[".substr(str replace("-","/",$myrow[createtime]), | 
0,10)."] </font>";?></td> | 
</tr> 
<tr> 

<td colspan="3"></td> 
</tr> 


we 
</table> 


问题 61 如何 定义 上 传 文件 的 名 称 ? 


问题 前 述 | 
在 上 传 文件 时 需要 指定 上 传 文件 路 径 , 但 面 对 大 量 用 户 上 传 的 不 同文 件 ， 难 免 会 有 重 名 的 | 
可 能 ， 这 样 就 会 造成 文件 替换 ， 从 而 导致 文件 丢失 ， 所 以 对 上 传 文件 的 名 称 进行 重新 定义 是 非 | 
常 有 必要 的 。 | 
专家 解答 

1. 解决 思路 

通过 strstr0) 函 数 对 上 传 文件 名 称 进行 截取 ， 获 取 上 传 文件 的 后 级 。 

通过 time0 函 数 获取 系统 当前 时 间 稚 ， 重 新 定义 上 传 文件 的 名 称 。 

应 用 move_uploaded file0 函 数 实现 文件 上 传 。 

在 执行 文件 上 传 之 前 ， 为 了 防止 潜在 的 攻击 对 原本 不 能 通过 脚本 交互 的 文件 进行 非法 管理 ， | 
应 用 is_uploaded_file0 函 数 判 断 指定 的 文件 是 否 是 通过 HTTP POST 上 传 的 , 如 果 是 则 返回 Trme。 | 

(1) is_uploaded_file0 函 数 ， 判断 指定 的 文件 是 否 是 通过 HTTP POST 上 传 的 。 其 语法 如 下 : | 

bool is_uploaded file ( string filename ) | 

参数 flename 必须 指定 类 似 于 $_FILES['filename']['tmp_name'] 的 变量 ， 不 可 以 使 用 从 客户 | 
端 上 传 的 文件 名 $_FILES['filename']['name']。 | 

通过 is_uploaded_file0 函 数 对 上 传 文件 进行 判断 ， 可 以 确保 恶意 用 户 无 法 欺骗 脚本 去 访问 | 
原本 不 能 访问 的 文件 ， 如 /etc/passwd。 | 

(2) move_uploaded file0 函 数 将 文件 上 传 到 服务 器 中 指定 的 位 置 。 如 果 成 功 返 回 True， 
否则 返回 False。 语 法 如 下 : 

bool move_uploaded file ( string filename, string destination ) 


0 UD 


人 ip 六 预 知道 的 300 个 问题 


| 参数 flename 指定 上 传 文件 的 临时 文件 名 ， 即 $_FILES[tmp_name]， 参 数 destination 指 文 

| 件 上 传 后 保存 的 新 路 径 和 名 称 。 如 果 参 数 fiename 不 是 合法 的 上 传 文件 , 则 不 会 出 现任 何 操作 ， 

”move_uploaded file0 将 返回 False; 如 果 参 数 flename 是 合法 的 上 传 文件 , 但 出 于 某 些 原因 无 法 
会 | 移动 , 则 也 不 会 出 现任 何 操 作 , move_uploaded file0 将 返回 False, 此 外 还 会 发 出 一 条 警告 信息 。 
A | (3) rand0 函 数 ， 使 用 其 生成 的 随机 整数 作为 上 传 文件 的 新 名 称 。 


2， 应 用 示例 
| 本 未 例 重新 定义 上 传 文件 的 名 称 ， 选 择 要 上 传 的 文件 ， 单 击 “ 提 交 ” 按 钮 ， 文 件 将 上 传 到 
| 程序 中 指定 的 本 地 路 径 下 的 upfile 文件 夹 中 ， 并 重新 定义 文件 名 称 。 运 行 结果 如 图 2.17 所 示 。 
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图 2.17 选择 上 传 文件 
其 关键 代码 如 下 : 


<2php 
这 !empty($_FILES[up_picture][name])){ /判断 上 传 内 容 是 否 为 空 
if($_FILES['up_picture']['error']>0){ /判断 文件 是 否 可 以 上 传 到 服务 器 
echo "上 传 错误 :"; 
Switch($_ FILES[up_picture]['error]){ 


ase 1: 
echo "上 传 文件 大 小 超出 配置 文件 规定 值 "; 
break:; 
ase 2: 
echo "上 传 文件 大 小 超出 表单 中 约定 值 "; 
break': 
Case 3: 
echo "上 传 文件 不 全 "; 
break: 
case 4: 
echo "没有 上 传 文件 "; 
break:; 
’ 
jelse{ 
if(!is_dir("./upfile/"){ // 判 断 指定 目录 是 否 存 在 
mkdir("./upfile/"): // 创 建 目录 


} 
$path='./upfile/'.time(.strstr($_FILES["up_picture']['name'],…");// 定 义 文件 名 称 和 存储 位 置 
ifis uploaded file($_ FILES[up_picture][tmp name])){/ 是 否 是 通过 HTTP POST 上传 的 
if(!move_uploaded file($_ FILES[up picture][tmp_ name'].$pathb)){ // 执 行 上 传 
echo "上 传 失 败 "; 
}else{ 
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echo "文件 "time0.$_ FILES[up_picture][mame']." 上 传 成 功 , 大 小 为 : ".$_FILES 


[up picture][size]; 
》 
}else{ 
echo "上 传 文件 ".$_ FILES[mp pictute][mame']." 不 合法 ! "; 
和 
} 
} 
> 
专家 点 评 


定义 上 传 文件 的 名 称 在 文件 上 传 程序 中 是 必 不 可 少 的 一 个 功能 , 通过 它 可 以 避免 重 名 文件 
被 替换 ， 但 这 可 能 会 导致 存在 很 多 相同 内 容 的 文件 。 | 


问题 62 ”如 何 通 过 字符 串 截 取 控 制图 像 验 证 码 的 输出 ? 


问题 阐述 | 
对 于 图 像 验 证 码 ， 相 信 大 家 并 不 陌生 ， 它 的 主要 作用 是 防止 用 户 恶 意 刷 新 注册 使 服务 器 系 | 
统 资源 流失 , 数据 库 匈 余 信 息 堆 积 。 图 像 验 证 码 的 出 现 避 免 了 相关 问题 发 生 的 可 能 性 , 创建 PHP | 
图 像 验证 码 需要 开启 GD2 函数 库 支持 ， 利 用 相关 的 图 像 函 数 创建 。 | 
专家 解答 
1， 解决 思路 | 
(1) 定义 数字 验证 码 的 取 值 范围 ， 通 过 rand0 函 数 获取 随机 数 ， 并 且 通 过 substr0 函 数 对 | 
随机 数 进行 截取 ， 同 时 将 截取 的 值 作为 验证 码 在 图 像 中 输出 。 | 
(2) 通 过 GD2 函数 库 中 的 函数 , 生成 图 像 验 证 码 , 其 应 用 的 GD2 函数 包括 : imagecreate0、 | 
imagecolorallocate()、imagefttext()、imageline()、magesetpixel()、imagejpeg()、imagedestroy() 控 | 
制 并 生成 验证 码 。 | 


2. 应 用 示例 
本 示例 生成 4 位 随机 验证 码 ， 其 关键 代码 如 下 : 
<?php 

header("Content-Type:text/html:charset=utf-8"): // 设 置 页 面 数 据 的 编码 风格 
header("Content-Type:image/jpeg"): // 告 知 浏览 器 输出 的 是 图 像 
$image=imagecreate(200,80): // 创 建 画 布 
$bgcolor=imagecolorallocate($image,230,230,230); /设置 背景 颜色 
$fontcolor=imagecolorallocate($image,0,0,0); // 设 置 字 体 颜 色 


$linecolor=imagecolorallocate($image,rand(0,200),rand(0,200),rand(0,200)); // 设 置 线 的 颜色 
$pixelcolor=imagecolorallocate($image,rand(0,200),rand(0,200),rand(0.200));// 设 置 噪点 的 颜色 


for($a=0:$a<10:$a+H){ 

S$rand.=dechex(rand(0,15)); // 获 取 10 个 十 六 进 制 数 
} 
for($b=0:$b<4:$b++){ 

S$str.=substr($rand,-$b,1)."” "; // 截 取 资 格 十 六 进 制 数 
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SD 
S$font="STCAIYUN.TTF"; /定义 字体 
imagefttext($image,30,rand(0,8),rand(0,40).rand(50,80).$fontcolor,$font,$str); ” // 输 出 字符 
for($c=0;$c<20;$c++){ 
imageline($image.rand(0,200),rand(0,80).rand(0,200).rand(0.80),$linecolor): /输出 20 条 线 
} 
for($d=0:;$d<1000;$d++){ 
imagesetpixel($image,rand(0,200),rand(0,80),$pixelcolon); /输出 1000 个 噪点 
中 
imagejpeg($image); /输出 图 像 
imagedestroy($image); 1/ 销毁 图 像 
?> 
运行 结果 如 图 2.18 所 示 。 
专家 点 评 


使 用 字符 串 截 取 控制 图 像 验证 码 的 输出 时 使 用 了 GD2 函数 库 , 要 图 2.18 图 像 验证 码 
使 用 该 函数 库 ， 需 要 修改 php.ini 文件 ， 将 “extension=php_gd2.dll” 选 项 前 面 的 分 号 去 掉 ， 然 后 
重新 启动 Apache 服务 器 。 


问题 63 ”如 何 打 乱 字 符 串 中 字符 的 顺序 ? 


问题 前 述 

打 乱 字符 串 的 字符 顺序 ， 比 较 典 型 的 应 用 是 实现 验证 码 的 生成 ， 例 如 ， 当 用 户 登 录 时 往往 
需要 输入 验证 码 ， 此 时 就 需要 将 字符 串 中 数字 的 顺序 随机 打 乱 ， 从 而 生成 固定 位 数 的 验证 码 。 
专家 解答 

1. 解决 思路 

随机 打 乱 字符 串 中 字符 的 顺序 ， 可 以 使 用 str_shuffle0 函 数 。 首 先 使 用 str_shuffle0 函 数 定 
义 随机 数 ， 然 后 使 用 substr0 函 数 对 随机 数 进行 截取 ， 最 后 使 用 for 循环 ， 循 环 输出 随机 数 ， 生 
成 数字 验证 码 。 

2. 应 用 示例 

下 面 使 用 str_shuffle() 函 数 创建 登录 页 面 中 的 数字 验证 码 ， 运 行 结果 如 图 2.19 所 示 。 


一 


用 户 : 轩 晶 科技 
sD: eeeeon 
站 : 可 5341 


Ra 


图 2.19 登录 页 面 中 的 验证 码 
关键 代码 如 下 : 


<?php 
Snum=str_shuffle("123456789"): 


$nums=substr($num.0.4); 
for($1=0;$1<4;$1++) { 

echo "<img src=images/".substr(strval($num),$1,1).".gif>"; 
1 


> 


问题 64 如何 对 字符 串 进 行 大 小 写 转换 ? 


问题 阐述 | 
网 站 中 有 很 多 数据 需要 具有 规则 性 ， 以 便于 管理 员 管 理 ， 但 为 了 使 用 户 输入 方便 ， 不 要 刻 | 
意 要 求 用 户 输入 大 写 或 小 写 ， 在 存储 数据 时 可 以 使 用 程序 控制 以 统一 大 写 或 小 写 的 方式 进行 | 
存储 。 | 
专家 解答 
1. 解决 思路 | 
对 字符 串 进行 大 小 写 转换 可 以 使 用 strtoupper 函数 和 strtolower 函数 。strtoupper 函数 可 以 | 
将 字符 串 转 换 为 大 写 ; strtolower 函数 可 以 将 字符 串 转换 为 小 写 。 | 


2. 应 用 示例 | 
下 面 将 用 户 填 写 的 图 书信 息 中 的 图 书 编号 全 部 以 大 写 形式 存储 ， 运 行 结果 如 图 2.20 所 示 。 | 


加 凶 景 入 的 结 票 


医书 名 称 :FHF300 癌 
编号 : ISBN-7-115-##* 
价格 49 
数量 : 10000 
产地 :明日 科技 


图 2.20 将 图 书 编号 转换 为 大 写 
关键 代码 如 下 : 


<2php session start(); 
ifissset($_ POST['Submit])){ 


$user=strtoupper($user):; // 将 商品 名 称 中 的 小 写字 符 转 换 为 大 写 
$number=strtoupper($number); // 将 编号 转换 为 大 写 
} 
?> 
说 明 : 
使 用 strtolower 函数 将 字符 串 转 换 为 小 写 的 使 用 方法 与 上 述 方 法 相同 。 
专家 点 评 


对 字符 串 进 行 大 小 写 转换 并 不 是 必要 功能 ， 用 户 可 以 根据 每 个 网 站 的 不 同 要 求 选择 使 用 。 | 
例如 ， 在 电子 商务 类 网 站 的 后 台 商 品 添加 页 面 中 ， 为 了 使 商品 的 型 号 以 统一 的 规格 显示 ， 经 常 | 
会 利用 大 小 写 转换 函数 ， 所 以 掌握 这 两 个 函数 是 很 有 必要 的 。 | 

a ke 
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问题 65 ”函数 strstr()、stristr()、strpos()、strrchr() 


? 
会 内 的 区 别 是 什么 ? 


问题 阐述 
| strstr()、stristr()、strpos()、strrchr() 函 数 的 功能 都 是 获取 一 个 指定 的 字符 串 在 另 一 个 字符 串 
”中 出 现 的 位 置 ， 那 么 ， 它 们 之 间 有 什么 区 别 呢 ? 
| 专家 解答 
| 1 解决 思路 
下 面 分 别 对 这 4 个 函数 进行 详细 介绍 : 
(1) stristr() 函 数 
| stristr() 函 数 用 于 获取 指定 字符 串 (A) 在 另 一 个 字符 串 (B)》 中 首次 出 现 的 位 置 到 (B) 字 
| 符 串 末尾 的 所 有 字符 串 。 该 函数 如 果 执行 成 功 则 返回 剩余 的 字符 串 ， 否 则 将 返回 False。 该 函 
| 数 不 区 分 大 小 写 , 如 果 要 在 区 分 字母 大 小 写 的 情况 下 进行 搜索 , 则 可 以 应 用 strstr0 函 数 , strstrO0 
| 函数 区 分 字母 的 大 小 写 。 
语法 格式 如 下 : 
string stristr(string haystack,string needle); 
stristr() 函 数 的 参数 说 明 如 表 2.7 所 示 。 
表 2.7 stristr() 函 数 的 参数 说 阴 


必要 参数 ， 指 定 要 查找 的 字符 串 
必要 参数 ， 指 定 要 查找 的 对 象 


(2) strstr0 函 数 
| strstr0 函 数 用 于 获取 一 个 指定 字符 串 在 另 一 个 字符 串 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字 
| 符 串 。 如 果 执行 成 功 ， 则 返回 剩余 字符 串 〈 存 在 相 匹配 的 字符 ); 如 果 没 有 找到 相 匹配 的 字符 ， 
| 则 返回 False。 
语法 格式 如 下 : 
string strstr ( string haystack, string needle) 
strstr0 函 数 的 参数 说 明 如 表 2.8 所 示 。 


表 2.8 strstr() 函 数 的 参数 说 明 


参数 说 明 
haystack “| 必要 参数 ， 指 定 从 哪个 字符 串 中 进行 搜索 
i 必要 参数 ， 指 定 搜索 的 对 象 。 如 果 该 参数 是 一 个 数值 ， 屠 么 将 搜索 与 这 个 数值 的 ASCIT 


值 相 匹 配 的 字符 


(3) strpos(0) 函 数 


strpos0 函 数 用 于 查找 指定 字符 串 〈A) 在 字符 串 〔B) 中 首次 出 现 的 位 置 。 返 回 值 为 字符 | 
了 。 | 


串 所 在 的 位 置 ， 如 果 失 败 则 返回 False。 本 函数 区 分 字母 的 大 小 和 
语法 格式 如 下 : 
int strpos(string haystack,string needle,int offset) 
strpos(0) 函 数 的 参数 说 明 如 表 2.9 所 示 。 
表 2.9 strpos() 函 数 的 参数 说 明 


参数 说 明 

haystack 必要 参数 ， 指 定 从 哪个 字符 串 中 开始 检索 字符 串 
needle 必要 参数 ， 指 定 要 检索 的 字符 串 

offset 必要 参数 ， 指 定 从 哪个 字符 串 开 始 检索 


(4) strrchrO) 函 数 
strrchr(O) 函 数 用 于 查找 指定 字符 串 〈A) 在 另 一 个 字符 串 〈B) 中 最 后 一 次 出 现 的 位 置 。 返 
回 从 字符 串 (B) 中 的 这 个 位 置 起 , 一 直到 字符 串 (B) 结束 的 所 有 字符 ; 如 果 失 败 则 返回 false。 
本 函数 区 分 字母 的 大 小 写 。 
语法 格式 如 下 : 
string strrchr(string haystack,string needle) 
strrchr() 函 数 的 参数 说 明 如 表 2.10 所 示 。 


表 2.10 strrchr() 函 数 的 参数 说 明 


参数 说 明 
haystack | 指定 从 哪个 字符 串 中 进行 搜索 

i 指定 要 搜索 的 字符 串 ， 如 果 要 搜索 的 是 一 个 数字 ， 那 么 将 搜索 与 该 数字 对 应 的 ASCII 值 相 
needle 


匹配 的 字符 
2. 应 用 示例 


下 面 分 别 用 strstrO0、stristr0、sttposO0 和 strrchr0 函 数 查找 同样 的 字符 串 ， 观 察 其 输出 结果 ， 
代码 如 下 : 


<?php 

$str = ‘mingrisoft(@WMingrisoft.com'; 

echo strstr($str, 'M).'<br/>': // 提 取 从 查找 字符 串 第 一 次 出 现 到 最 后 的 字符 串 ， 区 分 大 小 写 
echo stristr($str, 'M).'<br>': // 提 取 从 查找 字符 串 第 一 次 出 现 到 最 后 的 字符 串 ， 不 区 分 大 小 写 
echo sttpos($str'M).'<br/>': // 返 回 字符 串 第 一 次 出 现 位 置 ， 区 分 大 小 写 

echo strrchr($str, "M").'<br/>'; /提取 从 查找 字符 串 最 后 一 次 出 现 到 最 后 的 字符 串 ， 区 分 大 小 写 


?> 

运行 结果 为 : 
Mingrisoft.com 
mingrisoft@WMingrisoft.com 
11 
Mingrisoft.com 
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专家 点 评 
上 述 4 个 函数 虽然 相似 ， 但 都 有 各 自 不 同 的 用 处 ， 读 者 一 定 要 分 清 每 个 函数 实现 的 功能 。 


问题 66 站 和 switch 语句 有 何 区 别 ? 


| 问题 前 述 


过 和 switch 语句 有 何 区 别 ? 


专家 解答 


让 和 switch 语句 可 以 从 使 用 的 效率 上 来 进行 区 分 ， 也 可 以 从 实用 性 角度 去 区 分 。 如 果 从 使 
用 的 效率 上 进行 区 分 , 在 对 同一 个 变量 的 不 同 值 作 条 件 判 断 时 ， 使 用 switch 语句 的 效率 相对 更 
高 一 些 ， 尤 其 是 判断 的 分 支 越 多 越 明显 。 

如 果 从 语句 的 实用 性 的 角度 去 区 分 ， 那 么 switch 语句 肯定 不 如 让 条 件 语句 ， 可 以 说 ,让 条 
件 语句 是 实用 性 最 强 和 应 用 范围 最 广 的 语句 。 

在 程序 开发 的 过 程 中 ， 寺 和 switch 语句 的 使 用 应 该 根据 实际 的 情况 而 定 ， 不 要 因为 switch 
语句 的 效率 高 就 一 味 地 使 用 ,也 不 要 因为 站 语句 常用 就 不 应 用 switch 语句 。 用 户 应 根据 实际 的 
情况 ， 具 体 问 题 具体 分 析 ， 使 用 最 适合 的 条 件 语 句 。 一 般 情况 下 可 以 使 用 条 件 语句 ,但 是 在 
实现 一 些 多 条 件 的 判断 中 ， 特 别 是 在 实现 框架 的 功能 时 就 应 该 使 用 switch 语句 。 


问题 67 ”如 何 防止 页 面 出 现 中 文 乱 码 ? 


问题 阐述 
如 何 防 止 页 面 出 现 中 文 乱码 ? 
专家 解答 


在 编写 PHP 程序 时 ， 为 了 防止 页 面 出 现 乱 码 ， 要 求 PHP 文件 编码 、 网 页 页 面 编码 和 数据 
库 编 码 一 致 ， 如 果 采 用 UTF-8 编码 ， 则 上 述 3 项 内 容 都 要 设置 为 UTF-8 编码 ， 否 则 会 出 现 乱 
码 现象 那么 如 何 对 上 述 内 容 编码 进行 设置 呢 ? 设置 PHP 文件 编码 可 以 采用 UltraEdit、Edit Plus 
和 Zend Studio 等 编辑 器 ， 如 果 使 用 UltraEdit 设置 PHP 页 面 文件 编码 ， 则 可 以 在 另存 页 面 文件 
时 选择 格式 为 UTF-8， 如 图 2.21 所 示 。 

设置 网 页 的 页 面 编码 可 以 使 用 PHP 中 的 header0 函 数 或 在 HIML 的 Meta 标记 中 设置 。 如 
果 页 面 完 全 由 PHP 脚本 编写 ， 则 可 以 使 用 header0 函 数 设置 页 面 编码 。 设 置 方法 如 下 所 示 : 

header('content-type:text/html:; charset=UTF-8"); 


如 果 页 面 中 包含 规范 的 HIML 代码 , 则 可 以 在 HTML 的 Meta 标记 中 设置 页 面 编码 ,其 设 
置 方法 如 下 所 示 : 
<meta http-equiv="content-type" content="text/html; charset=UTF-8" /> 
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图 2.21 使 用 UltraEdit 另存 PHP 文件 为 UTF-8 编码 


完成 对 PHP 文件 及 页 面 编码 的 设置 后 ， 还 需要 设置 数据 库 编码 ， 一 般 需要 进行 两 处 设置 ， 
在 创建 数据 表 时 要 设置 字段 的 字符 编码 ， 其 设置 方法 如 图 2.22 所 示 。 | 


类 uO 长 度 剧 六 村 于 


图 2.22 设置 表 中 字段 的 字符 编码 | 

完成 上 述 设置 后 ,一 般 还 需要 在 程序 中 采用 编码 方式 设置 数据 库 字符 集 ， 其 设置 方法 可 以 
通过 SQL 语句 实现 ， 例 如 ， 通 过 如 下 语句 设置 数据 库 字 符 集 为 UTF-8。 
$conn->query("set names utf8"): | 

专家 点 评 
在 设置 数据 库 字符 编码 为 UTF-8 时 ， 应 为 set names utf8 而 非 set names utf-8， 注 意 不 要 多 

加 字符 “-”。 | 
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吾 吾 吾 于 于 于 于 于 于 于 于 芋 至 


羡 部 
王 王 


吾 吾 吾 吾 吾 吾 吾 吾 至 


如 何 打 印 指定 框架 中 的 内 容 ? 

如 何 使 用 框架 谋 套 技术 居中 显示 框架 页 ? 
如 何 为 文本 框 设置 只 读 属性 ? 

如 何 实 现 自动 计算 金额 ? 

如 何 实现 复 选 框 中 的 全 选 、 反 选 和 不 选 ? 
如 何 对 上 传 的 图 片 进行 预览 ? 


如 何 制作 在 页 面 右 下 角 弹 出 渐 显 的 广告 窗口 


判断 表单 动作 的 技巧 有 哪些 ? 

如 何 使 用 JS 脚本 获取 、 输 出 标签 内 容 ? 
如 何 控制 客户 端 COOKIE? 
COOKIE 应 用 的 注意 事项 有 哪些 ? 
SESSION 和 COOKIE 的 区 别 有 哪 些 ? 
如 何 使 用 SESSION? 

如 何 结束 所 有 SESSION? 

如 何 阻止 浏览 器 缓存 页 面 ? 

如 何 控 制 页 面 背景 的 打印 ? 

如 何 实现 文字 滑动 效果 ? 

如 何 实 现 自动 登录 ? 

如 何 统计 用 户 在 线 时 间 ? 

如 何 限 制 用 户 访问 网 站 的 时 间 ? 

如 何 掌 控 登 录用 户 的 权限 ? 

如 何 让 SESSION 永久 有 效 ? 
SESSION 购物 车 

headerO 函 数 都 有 哪些 功能 ? 


诬 


? 


世 


问题 前 述 
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问题 68 如何 打 印 指 定 框 架 中 的 内 容 ? 


在 网 页 中 实现 打印 功能 时 ， 打 印 的 是 整个 网 页 中 的 所 有 内 容 ， 但 很 多 时 候 我 们 起 要 打印 的 | 
只 是 整个 网 页 中 的 部 分 内 容 ， 该 如 何 实现 呢 ? 这 就 是 我 们 在 这 里 要 介绍 的 方法 一 一 使 用 框架 实 | 
现 网 页 中 部 分 内 容 的 打印 。 | 


专家 解答 
1. 解决 思路 | 
要 打印 页 面 中 指定 的 内 容 ， 可 以 利用 框架 中 各 部 分 相互 独立 的 特点 ， 将 要 打印 的 内 容 放置 | 


在 一 个 框架 中 ， 在 打印 该 内 容 时 就 不 会 牵涉 到 整个 页 面 的 内 容 了 。 其 具体 方法 如 下 : 
parent.mainFrame .fcous(): 
window.printO; 


mainFrame: 表示 框架 的 名 称 。 


打印 功能 。 


通过 parent.mainFrame.fcous() 方 法 获取 框架 焦点 , 然后 应 用 window 对 象 的 print0 方 法 实现 


2.， 应 用 示例 

运行 frame 框架 实现 页 面 内 容 的 部 分 打印 。 其 具体 步骤 如 下 : | 

(1) 首先 设计 contentphp 页 面 ， 应 用 include 调用 数据 源 文 件 ， 再 应 用 do...while 循环 语 | 
句 将 要 打印 的 内 容 显 示 在 该 页 面 中 。 


<2php 
include "conn/conn.php"; 


$sql-mysql query("select * from tb_user"); 
S$info=mysql_fetch_array($sq)); 


dof?> 


<tr> 


<td height="30"><?php echo $infofusemame]:?></td> 
<td><?php echo $info[tel];?></td> 
<td><?php echo $info[linker]:?></td> 
<td><?php echo $info[linktel]:?></td> 
<td><?php echo $info[email]:?></td> 
<td><?php echo $info[address]:?></td> 
</tr> 
<2php }while($info=mysql_fetch array($sqD))?> 


(2) 设计 index.php 页 面 ， 在 该 页 面 的 适当 位 置 添加 浮动 框架 (应 用 <iftame> 标 记 创 建 )， | 
命名 为 “content”， 并 将 该 浮动 框架 的 sre 属性 指定 为 步骤 (1) 中 创建 的 contentphp 文件 , 代 | 


码 如 下 : 


<iframe name="content" src="content.php" frameborder="0" width="100%" height="100%"></iframe> 
(3) 在 index.php 页 面 中 添加 “打印 ” 超 链接 ， 打 印 指定 浮动 框架 content 中 的 内 容 ， 代 | 
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| 码 如 下 
| <a href="#" onClick="parent.content focus():window.printO:"> 打 印 </a> 

| 其 运行 结果 如 图 3.1 所 示 ， 单 击 “ 打 印 ” 超 链接 ， 弹 出 打印 对 话 框 ， 进 行 相应 的 设置 后 ， 
| 即 可 打印 文本 框 中 的 内 容 。 


寺村 省 明日 科技 有 限 公司 
长春 市 er 叶子 经久 有 限 公司 
比 京 软 件 国有 限 责任 公司 


| 天津 华 遍 ew 有限 责任 公司 


取消 [EE 
图 3.1 打印 指定 框架 中 的 内 容 


问题 69 ”如 何 使 用 框架 嵌 套 技术 居中 显示 框架 页 ? 


| 问题 阐述 
| 使 用 框架 可 以 将 一 个 完整 的 页 面 分 割 成 不 同 的 窗口 ， 每 个 窗口 相互 独立 ， 且 每 个 窗口 中 的 
| 内 容 在 运行 时 互 不 干扰 。 虽然 将 页 面 分 割 成 多 个 窗口 在 某 些 时 候 较 为 实用 , 但 也 存在 一 些 问题 ， 
| 就 是 如 何 能 够 让 分 制 后 的 整个 页 面 居中 显示 ， 因 为 我 们 所 使 用 的 计算 机 显示 器 的 大 小 不 同 ， 在 
| 输出 内 容 时 会 随 着 显示 器 大 小 的 变化 而 变化 ， 为 了 让 网 页 更 加 美观 ， 必 须 保证 其 居中 输出 。 那 
| 么 如 何 才能 保证 框架 中 的 内 容 居中 显示 呢 ? 
专家 解答 

框架 是 指 网 页 的 各 部 分 是 相互 独立 的 页 面 ,并且 通 过 一 个 网 页 将 这 些 相互 独立 的 网 页 组 成 
| 一 个 完整 的 网 页 显示 在 浏览 器 中 ， 重 复出 现 的 内 容 被 固定 下 来 ， 当 每 次 浏览 者 发 出 对 网 页 的 请 
， 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 其 他 的 页 面 保持 不 变 。 使 用 框架 可 以 将 浏览 器 的 窗口 划分 
| 为 若干 个 子 窗口 ， 在 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 。 


1. 框架 网 页 的 基本 结构 


框架 网 页 的 基本 结构 如 下 : 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 
<title> 框 架 网 页 </title> 
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</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 
<body> 
对 不 起 ! 您 的 浏览 器 不 支持 框架 网 页 的 显示 ! 
</body> 
</noframes> 
</html> 
2. 设置 框架 集 的 属性 | 
框架 集 包 含 如 何 组 织 各 个 框架 的 信息 ， 可 以 通过 <framesef> 标 记 定义 。 框 架 是 按照 行 和 列 | 
来 组 织 的 ， 可 以 使 用 <frameset> 标 记 的 属性 对 框架 的 结构 进行 设置 。 | 
(1) 左右 分 割 窗口 属性 cols， 在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 其 语法 格式 如 下 : | 
<frameset cols="value,value,......"> | 
<frame> 
<frame> 
</frameset> 


value: 用 于 指定 各 个 框架 的 列 宽 ， 取 值 有 3 种 形式 : 像素 、 百 分 比 (%)》 和 相对 尺寸 (*)。 


说 明 : | 
如 果 将 cols 属性 设置 为 “*，*，*”， 则 表示 将 窗口 划分 成 3 个 等 宽 的 框架 ; 如 果 将 cols | 
属性 设置 为 “*，2*，3*”， 则 表示 左边 的 框架 占 窗口 宽度 的 1/6， 中 间 的 框架 占 窗口 宽度 的 1/3，| 
右边 的 框架 占 窗口 宽度 的 1/2。 | 


(2) 上 下 分 割 窗口 属性 rows, 在 生 直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 其 语法 格式 如 下 : 


<frameset rows="value,value,......"> 
<frame> 
<frame> 

</frameset> 


value: 用 于 指定 框架 的 行 高 ， 取 值 有 3 种 形式 : 像素 、 百 分 比 (%) 和 相对 尺寸 (*)， 设 | 
置 的 方式 与 cols 属性 类 似 。 | 
(3) 框架 边框 显示 属性 frameborder | 
该 属性 用 于 指定 框架 周围 是 否 显示 边框 ， 取 值 有 1 (显示 边框 , 默认 值 ) 和 0 (不 显示 边框 )。| 
(4) framespacing | 
该 属性 用 于 指定 框架 之 间 的 间隔 , 以 像素 为 单位 。 如 果 不 设 置 该 属性 , 则 框架 之 间 没 有 间隔 。 | 
(5) 指定 边框 宽度 属性 border | 
该 属性 用 于 指定 边框 的 宽度 ， 只 有 当 frameborder 属性 为 1 时 有 效 。 


3. 设置 框架 的 属性 | 
使 用 <frame> 标 记 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 、 是 否 包 含 滚动 条 以 及 在 框架 中 | 
显示 网 页 等 。 其 语法 格式 如 下 : | 
<frame src=" 文 件 " name=" 名 称 " scrolling=" 值 " [noresize] frameborder=-" 数 值 "> 
“ge L_ 
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&% 

参数 说 明 : 

@ name: 指定 框架 名 称 。 

@ src: 指定 在 框架 中 显示 的 网 页 文件 。 

@@ frameboder: 指定 框架 周围 是 否 显示 边框 ， 取 值 有 1 (显示 边框 ) 和 0 (不 显示 边框 )。 

@ noresize: 可 选 属性 ， 如 果 指 定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 。 

@ scrolling: 指定 框架 是 否 包 含 滚动 条 。 如 果 将 该 属性 设置 为 “yes”， 则 框架 包含 滚动 条 ; 
| 如 果 将 该 属性 设置 为 “no”， 则 表示 不 包含 滚动 条 ;如 果 将 该 属性 设置 为 “auto”， 则 在 需要 时 


J 


4. 应 用 示例 
在 使 用 框架 技术 设计 网 站 布局 时 ， 有 一 个 问题 是 必须 要 考虑 的 ， 那 就 是 浏览 器 的 分 辩 率 。 


如 果 使 用 1024*768 的 分 辩 率 制作 框架 ,那么 页 面 在 更 高 分 辩 率 的 显示 器 中 将 会 是 什么 样子 呢 ? 
| 这 就 需要 在 使 用 框架 设计 布局 时 增加 一 个 框架 嵌 套 技术 ， 使 框架 页 在 浏览 器 中 居中 显示 。 其 具 
| 体 方法 如 下 : 


<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 
<title> 使 用 框架 霸 套 技术 居中 显示 框架 页 </title> 
</head> 
<frameset rows="*" cols="*,630,*" frameborder="no" framespacing="0" border="0"> 
<frame sre="blank.php" name="BLFrame" scrolling="no" noresize> 
<frameset rows="55,*" cols="*" frameborder="no" border="0" framespacing="0"> 
<frame src="software_top.php" name="topFrame" scrolling="no" noresize> 
<frameset rows="*" cols="160,*" framespacing="0"> 
<frame src="software_left.php" name="leftFrame" scrolling="auto" noresize> 
<frame src="software_main.php" name="mainFrame" frameborder="no" scrolling="auto"> 
</frameset> 
</frameset> 
<frame src="blank.php" name="BRFrame" scrolling="no" noresize> 
</frameset> 
<noframes> 
<body> 
该 浏览 器 不 支持 框架 ， 无 法 正常 显示 !! 


在 此 框架 中 包括 4 个 子 页 面 路 径 :blank.php、software_top.php、software_left.php 和 software_ 
main.php。 其 运行 结果 如 图 3.2 所 示 。 无论 在 多 大 分 辨 率 的 显示 器 下 ， 其 都 能 够 正常 地 居中 显示 。 


专家 点 评 


这 里 介绍 的 方法 不 仅 应 用 了 框架 技术 ， 同 时 还 应 用 了 框架 嵌 套 技术 ， 使 网 页 无 论 在 多 大 分 
辨 率 的 显示 器 中 都 能 够 居中 显示 。 框 架 棋 套 技术 主要 就 是 在 划分 浏览 器 的 窗口 时 首先 对 窗口 进 
行 一 次 左右 的 分 割 ，cols 的 值 “cols="*,630,*"” 它 表示 该 框架 的 中 间 窗 口 为 630 像素 ， 且 左 
右 窗口 平分 。 其 中 中 间 窗 口 的 宽度 就 是 网 页 界面 的 实际 宽度 ， 然 后 ， 在 此 基础 上 再 进行 其 他 窗 
口 的 划分 。 
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问题 70 ”如 何 为 文本 框 设置 只 读 属 性 ? 


问题 阐述 


在 网 页 中 ， 有 些 文本 框 中 的 信息 是 不 允许 浏览 者 进行 修改 的 ， 即 用 户 只 有 浏览 的 权限 ， 而 
没有 修改 的 权限 ， 如 何 实现 此 功能 呢 ? | 


专家 解答 


对 文本 框 设置 只 读 属性 主要 是 对 文本 框 中 readonly 属性 的 运用 , 通过 它 设置 文本 框 的 只 读 
属性 ， 然 后 通过 onfocus 事件 调用 JavaScript 脚本 ， 当 用 户 单 击 已 经 设置 为 只 读 属 性 的 文本 框 | 
时 ， 弹 出 一 个 提示 对 话 框 。 | 

使 用 readonly 属性 实现 只 读 的 代码 如 下 : 

<input id=-"o" class="0" type="text" readonly value="0100521041'> | 

例如 ， 设 置 表单 中 输出 的 库存 数量 文本 框 为 只 读 属性 ， 并 且 通 过 value 属性 设置 文本 框 的 | 

默认 值 。 其 代码 如 下 : | 

<form action="" method="post"> 

商品 编号 : <input id="o" class="o" type="text" readonly value='0100521041'/><br> 

商品 名 称 : <input id="o"type="text" value=" 酸 牛奶 " onfocus="this.value="" /><br> 

库存 数量 : <input id="o" class="n"type="text" disabled="disabled" value="100 件 " /><br> 

管理 员工 : <input id="o"type="text" value="001" /> 

</form> 
专家 点 评 


在 使 用 readonly 时 ， 只 需 在 <input> 标 记 中 加 入 属性 名 称 即 可 。 


问题 71 如 何 实现 自动 计算 金额 ? 


问题 讲述 | 
在 物流 管理 系统 中 ,经常 需 要 计算 出 库 商品 的 金额 或 入 库 商品 的 金额 ， 若 以 手工 计算 比较 | 
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CT 
| 
麻烦 而 且 容易 出 错 ， 如 果 在 程序 中 实现 自动 计算 金额 功能 ， 则 可 以 减少 许多 不 必要 的 麻烦 。 
| 专家 解答 
了 要 实现 自动 计算 金额 , 主要 通过 触发 表单 控件 的 Blur 事件 ， 即 表单 控件 失去 焦点 时 ,实现 
医 网 | 自动 计算 金额 。 实 现 此 功能 的 JavaScript 代码 如 下 : 
$(".0") blur(functionO{ // 当 商品 单价 文本 框 失 去 焦点 
Var price = $(".0").valO|; 
var num = $(".n").val(|); 
Var amount = price * num; 
$(".e").val(amount); 
1 
例如 ， 通 过 jQuery 脚本 自动 计算 金额 ， 并 设置 输出 金额 的 文本 框 为 内 读 ， 其 代码 如 下 : 
<form action="" method="post" name="form1"> 
商品 编号 ，<input id="o" type="text" readonly value= 酸 牛奶 /><br> 
商品 单价 : <input id="o" type="text"class="0" value="0.0"/><br> 
库存 数量 : <input id="o" class="n"type="text" value="35" readonly /><br> 
金额 总 计 : <input id="o" class="e"type="text" value="0.0" /> 


</form> 
运行 结果 如 图 3.3 所 示 。 
| 图 3.3 自动 计算 金额 
| 专家 点 评 
| 用 于 当 Blur 事件 ( 即 失去 焦点 事件 ) 发 生 时 执行 JavaScript 代码 。 适 用 对 象 包括 Button、 


| Checkbox, FileUpload、 Layer、 Password、 Radio、 Reset, Select, Submit、 Text、 Textarea 和 Window。 
问题 72 ”如何 实现 复 先 框 中 的 全 选 、 反 选 和 不 选 ? 

| 问题 阐述 

| 复 选 框 常 被 应 用 在 数据 管理 系统 中 ， 如 多 数据 的 删除 等 。 如 果 想 要 删除 全 部 的 数据 ， 手 动 

| 选择 每 条 数据 前 面 的 复 选 框 , 显得 过 于 麻烦 , 那么 该 如 何 实现 复 选 框 的 全 选 、 反 选 和 不 选 呢 ? 
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1. 解决 思路 
复 选 框 的 全 选 、 反 选 和 不 选 主要 应 用 JavaScript 脚本 ， 通 过 自 定义 函数 完成 。 | 
(1) 在 通过 JavaScript 脚本 中 的 自 定义 函数 完成 复 选 框 的 全 选 、 反 选 和 不 选 功能 中 , 应 用 | 
的 第 一 个 方法 是 ，getElementsBByTagName， 获 取 指 定 标 签 的 名 称 ， 返 回 值 为 一 个 包含 标签 信息 | 
的 object。 
(2) 根据 getElementsByTagName 标签 返回 的 对 象 ， 判 断 标签 类 型 (type) 的 值 是 否 是 | 
checkbox。 | 
(3) 当 标 签 类 型 ype 的 值 为 checkbox 时 ， 为 标签 中 的 checked 赋值 。 当 checked 的 值 为 | 
true 时 ， 为 它 赋值 为 false; 当 checked 的 值 为 false 时 ， 为 它 赋值 为 true。 | 
(4) 在 页 面 中 通过 script 标签 调用 js 文件 夹 下 的 文件 ， 应 用 onClick 事件 调用 自 定义 函 | 
数 ， 实 现 全 选 、 反 选 和 不 选 功 能 。 
2. 应 用 示例 
在 本 示例 中 实现 复 选 框 的 全 选 、 反 选 和 不 选 功 能 。 其 具体 步骤 如 下 : 
(1) 创建 js 文件 夹 ， 编 写 regjs 脚本 文件 。 然 后 在 regjs 中 编写 自 定义 函数 ， 实 现 全 选 、 
反选 和 不 选 功能 。regjs 文件 的 关键 代码 如 下 : 


function uncheckAll(form!l ,status) { // 不 选 | 
var elements = forml.getElementsByTagName(input); /获取 input 标签 | 
for(var i=0: i<elements.length; i++){ // 根 据 标签 的 长 度 执 行 循环 | 
if(elements[i].type =— 'checkbox'’) { // 判 断 对 象 中 元 素 的 类 型 | 

这 elements[i].checked 一 true){ /判断 当 checked 的 值 为 tue 时 | 
elements[i].checked=false: /为 checked 赋值 为 false | 

} | 

} 

1 
function checkAll(form!l,status){ // 全 选 | 


Var elements = form!l .getElementsByTagName('input'); 
for(var i=0; i<elements.length; i++){ 
if(elements[i].type =—= 'checkbox'") { 
这 elements[i].checked 一 false){ 
elements[i].checked=true; | 
} | 
} | 
} | 
} | 
function switchAll(forml,status) { /反选 | 
Var elements = forml .getElementsByTagName('input'); | 
for(var i=0; i<elements.length; i++){ | 
这 elements[i].type =— 'checkbox'"){ | 
这 elements[i].checked 一 true){ | 
elements[i].checked=false; | 
}else ifelements[i].checked- 一 false){ 
elements[i].checked=true: 
} 
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(2) 创建 index.php 页 面 ， 输 出 会 员 信息 ， 并 添加 图 像 按钮 ， 通 过 onClick 事件 调用 JavaScript 
自 定义 函数 实现 全 选 、 反 选 、 不 选 和 删除 的 功能 。 其 关键 代码 如 下 : 


<script language="javascript" src="js/reg.js"></script> 
We <form method="post" name="form1" id="form1" action="index_ok.php"> 


<tr> 
<td width="62"align="center"><input name= "conn id[]" type="checkbox" id="conn id[]"value="1"/></td> 
<td width="242"><?php echo $myrow[id];?></td> 
<td width="243"><?php echo $myrow[user];?></td> 
<td width="485"><?php echo $myrow[pass]:?></td> 
<td width="485"><?php echo $myrow[dates];?></td> 
</tr> 
<tr> 
<td colspan="5" align="center"><!-- 通 过 onClick 事件 调用 自 定义 函数 ， 执 行 相应 的 操作 --> 
<img src="images/bg_19-20.jpg" onclick="checkAll(forml,status)" width="62" height="25" /> 
<img strc="images/bg_14-14.jpg" onclick="switchAll(form!l,status)"” width="62" height="25" /> 
<img src="images/bg_07-08.jpg" width="62" height="25" onclick="uncheckAll(form!,status)" /> 
<input type="image" name="imageField" src="images/bg_14.jpg" 人 > 
| </td> 
| </tr> 
</form> 


复 选 框 的 全 选 、 反 选 和 不 选 主要 应 用 JavaScript 脚本 ， 通 过 JavaScript 脚本 中 的 自 定义 函 
数 ， 判 断 复 选 框 中 checked 的 值 。 全 选 就 是 为 checked 赋值 为 tue， 不 选 则 是 为 checked 赋值 为 
false， 而 反选 则 是 当 checked 的 值 为 tue 时 ， 为 它 赋值 为 false; 当 checked 的 值 为 false 时 ， 为 
它 赋值 为 tue。 


| 问题 73 如何 对 上 传 的 图 片 进行 预览 ? 


| 问题 阐述 


在 上 传 图 片 时 ,往往 会 因为 没有 事先 查看 图 片 而 传 错 图 片 ， 如 果 可 以 实现 上 传 图 片 的 预览 
功能 ， 就 不 会 出 现 传 错 图 片 的 问题 了 。 那 么 如 何 实现 上 传 图 片 的 预览 功能 呢 ? 


专家 解答 


实现 上 传 图 片 的 预览 功能 应 用 的 是 jQuery 框架 中 的 方法 , 直接 读 取 上 传 图 片 , 并 且 将 读 取 
| 的 结果 输出 ， 同 时 还 判断 上 传 图 片 是 否 为 空 。 其 方法 存储 于 injs 脚本 文件 中 ， 代 码 如 下 : 


| $(document) .ready(function|){ 

| S$("#two").click(functionO{ // 单 击 预 览 图 片 

| var value = $("#one").val():; // 取 得 上 传 地 址 

| ifvalue — ""){ /判断 地 址 是 否 为 空 
| alert(" 地 址 为 空 "); 

| }else{ 
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$(".n").append("<img id='five' src="+value+">"); // 显 示 图 片 
$("#two").click(functionO{ 
$(".n").emptyO: 
$(".n").append("<img id='five' sre="+value+">"); 
D; 
9 
D); 
$("#three").click(functionO{ // 显 示 图 片 地 址 
var value = $("#one").val|; 
if(value — "){ 
alert(" 地 址 为 空 "); 
}else{ 
alert(value): 
} 


D; 
$("#four").click(functionO{ /清除 文本 框 信息 
$(".e").emptyO; 
$(".e").append("<input class='one' id='one' type='file'’>"); 
D; 
)); 
创建 index.php 文件 ， 引 入 CSS 样式 和 injs 文件 以 及 jQuery 库 文件 jquery-1.3.2js。 创 建 
form 表单 ， 设 置 文件 域 属性 ， 添 加 预览 图 片 、 图 片 地 址 和 清除 的 按钮 。 实 现 图 片上 传 和 预览 的 
功能 ， 其 代码 如 下 : 


<table width="600" height="450" align="center" background="pic/bg.jpg"> 
<tr> 
<td align="center"><div class="n"></div> 
<table align="center"> 
<tr> 
<td> 
<span class="0"> 
<span class="e"><input class="one" id="one" type="file"></span> 
<input class="one" id="two" type=button value=" 预 览 图 片 "> 
<input class="one" id="three" type=button value=" 图 片 地 址 "> 
<input class="one" id="four" type=button value=" 清 除 " > 
</span> 
</td> 
</t> 
</table> 
</td> 
</t> 
</table> 


运行 结果 如 图 3.4 所 示 。 
专家 点 评 
jQuery 框架 中 的 empty0 方 法 删除 匹配 的 元 素 集合 中 所 有 的 子 节点 。 其 语法 如 下 : 
S$('class/id").emptyO:; 


参数 说 明 : 
class/id: 表示 类 名 或 id 名 称 。 
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图 3.4 上 传 预览 图 片 


问题 74 如何 制作 在 页 面 右 下 角 弹 出 渐 显 的 广告 窗口 ? 


问题 前 述 


广告 作为 网 站 最 大 的 一 利 手段 ， 任 何 网 站 都 不 可 能 将 其 省 略 ， 那 么 ， 如 何 才 能 实现 网 站 既 


可 以 从 广告 
专家 解答 


P 获 利 ， 又 能 使 用 户 流畅 地 阅读 网 站 信息 呢 ? 


从 首页 右 下 角 弹 出 广告 窗 体 应 用 的 是 JavaScript 脚本 ， 通 过 window 对 象 在 HTML 中 重新 
打开 一 个 窗口 ， 应 用 open 方法 控制 打开 的 文件 及 窗口 的 大 小 ， 通 过 screen 方法 控制 窗口 在 页 


面 中 的 位 置 。 


其 具体 方法 的 代码 如 下 : 


<script language="javascript"> 
var newformW=300; 
var newformH=180; 
function ppO{ 
Var T=window.screen.width-newformW; 
Var L=window.screen.height+newformH:; 
var name=window.open("advertise.htm","","width="+newformW+",height="+newformH+",top= 
"+T+",left="+L); 


1 
ppO; 


</script> 
运行 结果 如 图 3.5 所 示 。 
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图 3.5 右 下 角 渐 显 广告 
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专家 点 评 
window 对 象 打开 窗口 详解 。 
使 用 window 对 象 打开 窗口 的 语法 格式 如 下 : | 
windowVar=window.open(url,windownamel[,location]):; | 食 和 


参数 说 明 : 
@ windowVar: : 当前 打开 窗口 的 句柄 。 如 果 open0 方 法 成 功 ， 则 windowVar 的 值 为 一 个 
window 对 象 的 句柄 ， 否 则 windowVar 的 值 是 一 个 空 值 。 | 
@ url: 目标 窗口 的 URL。 如 果 URL 是 一 个 空 字符 串 ， 则 浏览 器 将 打开 一 个 空白 窗口 ， 同 | 
时 允许 用 write0 方 法 创建 动态 HTML。 | 
windowname: window 对 象 的 名 称 。 
@ location: 对 窗口 属性 进行 设置 。 


问题 75” 判断 表单 动作 的 技巧 有 哪些 ? 


问题 阐述 | 
表单 可 以 通过 同一 个 程序 来 分 配 应 该 要 处 理 的 动作 ， 在 表单 中 有 不 同 的 逻辑 ， 应 如 何 判 别 
使 用 者 按 下 的 按钮 呢 ? | 
专家 解答 
要 解决 这 个 问题 很 简单 ， 只 要 通过 提交 按钮 的 name 就 可 以 知道 了 ， 表 单 在 提交 出 去 时 ， | 
只 有 按 下 的 submit 类 型 的 按钮 才 会 被 送 到 表单 数组 去 , 所 以 只 要 判断 按钮 的 值 就 可 以 知道 使 用 | 
者 按 下 哪 一 个 按钮 ， 以 如 下 表单 为 例 : | 
<form id="forml1" name="forml" method="post" action=""> 
<input type="submit" name="btn" value="a" /> 
<input type="submit" name="btn" value="b" /> 
</form> 
当 使 用 者 按 下 “a” 按 钮 时 ，btn=a; 按 下 “b” 按 钮 时 ，btn=b。 
另外 ， 也 可 以 通过 提交 按钮 的 名 字 Cname) 进行 判断 ， 见 如 下 代码 : 


<form id=-"fomml" name="forml" method="post" action=""> 
<input type="submit" name="a" value=" 提 交 A" /> 

<input type="submit" name="b" value=" 提 交 B" /> 

</form> 


这 样 ， 只 要 POST/GET 的 参数 中 有 a 或 b， 就 可 以 知道 按 下 的 是 哪个 按钮 了 。 
问题 76 ”如 何 使 用 JS 脚本 获取 、 输 出 标签 内 容 ? 
问题 前 述 


如 何 获取 、 更 改 表 单元 素 值 和 特定 标签 内 容 ? | 
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专家 解答 
| 使 用 JS 脚本 获取 页 面 内 容 的 方式 主要 有 两 种 ， 第 一 种 方式 是 通过 表单 获取 表单 元 素 的 
会 内 | value 值 。 格 式 为 : 
~ 表单 名 称 .元 素 名 value; 
该 方式 只 能 获取 表单 中 的 元 素 值 ， 对 于 其 他 标签 元 素 则 无 能 为 力 。 
| “第 二 种 方式 是 通过 id 名 来 获取 页 面 中 任意 标签 的 内 容 。 格 式 为 : 


document.getElementById(“id’). value; 
document.getElementById (“id’).innerText; 


使 用 第 二 种 方式 时 要 注意 ， 标 签 的 id 名 必须 存在 且 唯 一 ， 否 则 就 会 出 现 错误 。 
当 为 标签 内 容 赋值 时 ， 使 用 如 下 格式 : 
id.innerHTML = 要 显示 的 内 容 ': 


问题 77 ”如何 控制 客户 端 COOKIE? 


问题 阐述 i 阮 多 |m 安 | 还 招 | 有 车 | wi | 

| 如 何 控制 客户 端 COOKIE? [LS 
专家 解答 :| : 测 遇 SETasvo 
| 本 下 ni 
| 对 COOKIE 的 控制 属于 正 浏 览 器 Intemet 属 性 中 隐 J 

| 私 的 内 容 。 操作 方法 如 下 : 站). .| 导入 加 | BW. | _ MD 

| (1) 选择 正 浏览 器 ， 单 击 鼠 标 右键 ， 在 弹出 的 快 wun pe. 

| 捷 菜单 中 选择 “属性 ”命令 ， 在 弹出 的 Internet 属性 对 5 PLE | 
| 话 框 中 选择 “隐私 ”选项 卡 ， 如 图 3.6 所 示 。 

| (2) 移动 图 3.6 中 的 滑 块 ， 设 置 Internet 区 域 的 | 
| 陷 私 。 图 3.6 控制 客户 端 COOKIE 


(3) 设置 完成 后 ， 单 击 “ 应 用 ”按钮 ， 保 存 设置 。 


问题 78 ”COOKIE 应 用 的 注意 事项 有 哪些 ? 


| 问题 阐述 
| 在 学 习 及 使 用 COOKIE 时 ， 都 应 该 注意 哪些 事项 呢 ? 
| 专家 解答 


1 牢记 setcookie() 通 数 的 特性 
| 在 使 用 setcookie(0) 函 数 创建 COOKIE 之 前 ,不 能 有 任何 HTML (即使 是 空格 、 空 行 也 不 可 
| 以 ) 输出, 这 点 与 session_start 0 函数 类 似 。 如 果 犯 了 这 个 错误 , 那么 得 到 的 将 是 一 堆 错误 代码 。 
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2. 为 什么 8 COOKIE 在 当前 页 获取 不 到 COOKIE 值 | 
通过 setcookie0 函 数 创建 COOKIE 后 ， 在 当前 页 应 用 $_COOKIE 获取 不 到 COOKIE 的 值 ，| 
而 必须 是 在 刷新 或 到 达 下 一 页 面 后 才能 获取 到 COOKIE 值 。 因 为 setcookie() 函 数 执行 后 ， 向 客 | 
户 端 发 送 一 个 COOKIE， 如 果 不 刷 新 或 浏览 下 一 个 页 面 ， 客 户 端 就 不 能 将 COOKIE 送 回 。 | 
3. COOKIE 会 一 直 伴随 网 站 中 的 每 个 请 求 ， 直 到 COOKIE 过 期 
浏览 器 创建 一 个 COOKIE 后 ， 对 于 每 一 个 针对 该 网 站 的 请 求 ， 都 会 在 Header 中 带 着 这 个 | 
COOKIE， 而 且 浏览 器 会 这 样 一 直 发 送 ， 直 到 COOKIE 过 期 为 止 。 不 过 ， 对 于 其 他 网 站 的 请 求 | 
COOKIE 是 绝对 不 会 跟着 发 送 的 。 | 
4. 使 用 COOKIE 的 限制 | 
(1) 如 果 COOKIE 不 设置 过 期 时 间 ， 那 么 它 的 生命 周期 就 是 浏览 器 会 话 的 期 间 ， 只 要 关 | 
闭 浏览 器 , COOKIE 就 消失 , 这 是 会 话 COOKIE, 它 一 般 不 保存 在 硬盘 上 , 而 是 保存 在 内 存 中 。) 
(2) 如 果 COOKIE 设置 过 期 时 间 ， 那 么 浏览 器 就 会 把 COOKIE 保存 到 硬盘 中 ， 再 次 打开 | 
浏览 器 时 依然 有 效 ， 直 到 它 过 期 为 止 。 | 
(3) 不 是 所 有 的 浏览 器 都 支持 COOKIE。 | 
(4) COOKIE 数据 是 以 明文 的 形式 存储 于 客户 端的 计算 机 中 ， 不 适合 保存 敏感 的 、 未 加 | 
(5) COOKIE 存储 数据 量 有 一 定 的 限制 。 一 个 浏览 器 允许 最 多 存储 300 个 COOKIE 文 | 
件 ， 而 且 每 个 COOKIE 文件 支持 最 大 容量 为 4KB; 每 个 域名 最 多 支持 20 个 COOKIE， 如 果 达 | 
到 限制 ， 浏 览 器 会 自动 随机 删除 COOKIE。 | 


问题 79 SESSION 和 COOKIE 的 区 别 有 哪 些 ? 


问题 阐述 

SESSION 和 COOKIE 的 区 别 有 哪 些 ? 
专家 解答 

SESSION 和 COOKIE 最 大 的 区 别 是 : | 

SESSION 将 信息 保存 在 服务 器 上 ， 并 通过 一 个 SESSION ID 来 传递 客户 端的 信息 。 服 务 | 
器 在 接收 到 SESSION_ID 后 根据 这 个 ID 来 提供 相关 的 SESSION 数据 .COOKIE 是 将 所 有 的 信 | 
息 以 文本 文件 的 形式 保存 在 客户 端 ， 并 由 浏览 器 进行 管理 和 维护 。 | 
因为 SESSION 数据 存储 于 服务 器 端 ， 所 以 远程 用 户 无 法 对 SESSION 数据 进行 修改 。 而， 
COOKIE 数据 存储 于 客户 端 , 很 容易 被 获取 和 自 改 。 所 以 SESSION 的 安全 性 要 比 COOKIE 高 。 | 
专家 点 评 | 

对 于 SESSION 和 COOKIE 的 选择 ， 只 有 从 实际 的 需求 出 发 ， 选 择 最 适合 的 技术 才能 够 开 | 
发 出 最 理想 的 Web 程序 。 | 


“人 3 
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问题 80 如何 使 用 SESSION? 


网 -问题 曾 述 


在 Web 程序 中 ， 如 何 使 用 SESSION 呢 ? 


专家 解答 


| SESSION 被 译 成 中 文 为 “会 话 ”， 其 本 义 是 指 有 始 有 终 的 一 系列 动作 /消息 ， 如 打 电 话 时 从 
| 拿 起 电话 拨号 到 挂 断 电话 这 中 间 的 一 系列 过 程 可 以 称 之 为 一 个 SESSION。 

| 在 计算 机 专业 术语 中 , SESSION 是 指 一 个 终端 用 户 与 交互 系统 进行 通信 的 时 间 间 隔 , 通常 
| 指 从 注册 进入 系统 到 注销 退出 系统 之 间 所 经 过 的 时 间 。 其 工作 原理 如 图 3.7 所 示 。 

| 如 图 3.7 所 示 ， 当 登录 网 站 时 ， 启 动 SESSION 会 话 ， 在 服务 器 中 随机 生成 一 个 唯一 的 
| SESSION ID， 这 个 SESSION ID 在 本 次 登录 结束 之 前 在 页 面 中 一 直 有 效 。 当 关闭 页 面 或 执行 
| 注销 操作 后 ， 这 个 SESSION ID 会 在 服务 器 中 自动 注销 。 当 重新 登录 此 页 面 时 ， 会 再 次 生成 一 
| 个 随机 且 唯 一 的 SESSION ID。 


二 
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图 3.7 SESSION 工作 原理 


| (1) 使 用 SESSION 前 ， 必 须 先 调用 session_start(0) 函 数 ， 该 函数 主要 用 于 初始 化 一 个 新 的 
| SESSION， 语 法 格式 如 下 : 
| boolean session start(void) 

session start() 函 数 的 参数 为 空 。 
| (2) 会 话 变量 被 启动 后 ， 全 部 保存 在 数组 $ SESSION 中 。 通 过 数组 $ SESSION 创建 一 个 
| 会 话 变量 很 容易 ， 只 要 直接 给 该 数组 添加 一 个 元 素 即 可 。 

例如 ， 启 动 会 话 ， 创 建 一 个 SESSION 变量 并 赋予 空 值 ， 代 码 如 下 : 


<?php 

session start(; // 启 动 SESSION 

$_SESSION["user"] = null: // 声 明 一 个 名 为 user 的 变量 ， 并 赋 空 值 
?> 
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专家 点 评 


SESSION 在 Web 技术 中 占有 非常 重要 的 地 位 。 由 于 网 页 是 一 种 无 状态 的 连接 程序 ， 所 以 | 
无 法 获取 用 户 的 浏览 状态 。 因 此 必须 通过 SESSION 记录 用 户 的 有 关 信息 ， 以 供用 户 再 次 以 此 | 
身份 对 Web 服务 器 提供 要 求 时 做 确认 。 例如， 在 电子 商务 网 站 中 ， 通 过 SESSION 记录 用 户 登 | 
录 的 信息 ， 判 断 用 户 的 访问 权限 ， 通 过 SESSION 实现 购物 车 功能 ， 记 录用 户 所 购买 商品 。 | 5 
Ez 


问题 81 如何 结束 所 有 SESSION? 


问题 阐述 
SESSION 使 用 完成 后 ， 如 何 结束 SESSION 呢 ? 
专家 解答 
删除 会 话 有 3 种 方式 : 删除 单个 会 话 、 删 除 多 个 会 话 和 结束 当前 会 话 。 
删除 单个 会 话 
删除 会 话 变量 与 删除 数组 中 的 指定 元 素 相 同 ， 都 可 以 通过 unset0 函 数 完成 。 语 法 如 下 : 
unset ($_ SESSION[user] ) : | 
其 参数 是 $_ SESSION 数组 中 的 指定 元 素 ， 该 参数 不 可 以 省 略 。 通 过 unsetO 函 数 一 次 删除 | 
数组 中 的 一 个 元 素 ; 如 果 通 过 unset(O) 函 数 一 次 注销 整个 数组 (unset($_ SESSION))， 那 么 会 禁 | 
止 整个 会 话 功能 ， 而 且 无 法 将 其 恢复 ， 用 户 也 不 能 再 注册 $_SESSION 变量 。 所以， 如果 读者 要 | 
删除 多 个 或 全 部 会 话 ， 则 可 以 采用 下 面 的 两 种 方式 。 | 
回 ”删除 多 个 会 话 
如 果 要 一 次 注销 所 有 的 会 话 变量 ， 则 可 以 将 一 个 空 的 数组 赋值 给 $_SESSION， 代 码 如 下 : 
$_SESSION = array0 ; 
回 ”结束 当前 会 话 | 
如 果 整 个 会 话 已 经 结束 , 那么 就 可 以 使 用 session_destroy0 函 数 结束 当前 的 会 话 。 语 法 如 下 : | 
session_destroy(: | 
该 函数 清空 会 话 中 的 所 有 资源 ， 彻 底 销毁 SESSION。 


问题 82” 如何 阻止 浏览 回 组 存 页 面 ? 


问题 前 述 


如 果 “ 信 息 及 时 ”对 站 点 至 关 重 要 ， 并 且 希 望 过 时 的 信息 不 会 再 被 看 到 ， 就 需要 知道 如 何 | 
从 一 开始 就 阻止 浏览 器 和 代理 服务 器 缓存 页 面 ? 


专家 解答 
解决 这 个 问题 有 两 个 可 用 的 方法 ， 即 使 用 HIML meta 标签 和 使 用 HTTP 头 。 
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1. 使 用 HTML meta 标签 
阻止 缓存 页 面 最 常用 的 方法 是 使 用 HTML meta 标签 : 
| <meta http-equiv="expires" content="Wed,13 Jul 2011 05:00:00 GMT"/> 
食 放 | | <meta http-equiv="pragma" content="no-cache"/> 
ey 已 经 传递 到 Expires meta 标签 的 插入 日 期 告诉 浏览 器 ， 缓 存 的 页 面 找 贝 总 是 过 期 的 。 遇 到 
te | 此 标签 后 ， 浏 览 器 通常 不 会 缓存 页 面 。 虽 然 Pragma: no-cache meta 标签 并 不 是 保证 ， 但 它 却 是 
| 大 多 数 网 络 浏览 器 遵从 的 受到 良好 支持 的 约定 。 
| 2. 使 用 HTTP 头 
| 阻止 缓存 页 面 更 好 的 办 法 是 应 用 header0 函 数 ， 使 用 HTTP 本 身 协议 生成 与 上 面 等 价 的 两 
| 个 HTML meta 标签 : 
| <?php 
header(Expires:Wed,13 Jul 2011 05:00:00 GMT"); 
header(Pragma:no-cache); 
?> 


可 更 进一步 使 用 HITP1.1 的 浏览 器 所 支持 的 Cache-Control 头 : 


<2php 

header(Expires:Wed,13 Jul 2011 05:00:00 GMT"); 
header(Cache-Control: no-store,no-cache,must-revalidate’); 
header(Cache-Control: post-check=0,pre-check=0',FALSE); 
header('Pragma:no-cache'); 

?> 


| 专家 点 评 


虽然 使 用 Expires meta 标签 是 一 个 很 好 的 办 法 ， 但 是 随 之 而 来 的 有 两 个 问题 : 
| 第 一 ， 浏 览 器 必须 下 载 页 面 以 读 取 meta 标签 。 当 浏览 器 第 一 次 请 求 页 面 时 ， 如 果 不 存 在 
| 标签 ， 则 浏览 器 将 会 继续 全 然 不 知 并 保持 原文 件 的 缓存 拷贝 。 
| 缓存 网 络 页 面 的 代理 服务 器 ， 如 那些 共有 的 网 络 服务 器 提供 商 〈IPS)， 一 般 不 会 自己 读 取 
| HTML 文档 。 网 络 浏览 器 可 能 知道 它 不 应 该 缓存 页 面 ,但 在 浏览 器 和 网 络 服务 器 之 间 的 代理 服 
| 务 器 可 能 不 知道 一 一 它 将 继续 向 客户 端 传递 同样 过 时 的 页 面 。 
| 第 二 , 使 用 HTTP 协议 来 阻止 页 面 缓 存 从 根本 上 保证 了 没有 网 络 浏览 器 或 中 间 代 理 服务 器 
| 缓存 页 面 ， 这 样 访问 者 将 总 能 接受 到 最 近 的 内 容 。 实 际 上 ， 应 该 实现 的 第 一 个 头 应 是 其 自身 ， 
， 这 是 确保 页 面 不 被 缓存 的 最 好 方法 。Cache-Control 和 Pragma 头 只 是 为 了 增加 一 定 程度 的 保险 
| 而 添加 的 。 虽 然 它们 不 能 在 所 有 浏览 器 或 代理 上 工作 ， 但 是 Cache-Control 和 Pragma 头 将 捕获 
| Expires 头 不 像 预 期 那些 工作 的 一 些 情形 ， 例 如 ， 客 户 端 计算 机 日 期 设置 不 正确 。 


问题 83 ”如何 控制 页 面 背 景 的 打印 ? 


| 问题 阐述 
如 何 控制 页 面 背景 的 打印 呢 ? 
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专家 解答 


在 进行 Web 打印 时 ， 可 以 通过 以 下 操作 控制 Eire 
是 否 打印 背景 颜色 和 图 像 。 操 作 步 又 如 下 : 
(1) 在 下 窗口 中 选择 “工具 ”一 “Intemet Ee 
选项 ”命令 。 ey J 
(2) 在 弹出 的 “Intemet 选项 ”对 话 框 中 选择 
“高 级 ”选项 卡 ， 在 “打印 ”列表 中 设置 “打印 背 有 
景 颜色 和 图 像 ”前 面 的 复 选 框 是 否 勾 选 ， 如 图 3.8 en 
所 示 。 Gamws! wuesl 
(3) 如 果 勾 选 ， 则 代表 打印 背景 颜色 和 图 像 ， a 
否则 将 不 打印 背景 颜色 和 图 像 。 i i 
问题 84 ”如 何 实现 文字 滑动 效果 ? 
问题 阐述 


大 家 可 以 看 到 ， 在 很 多 网 站 中 的 公告 栏 信息 等 都 采用 文字 滑动 的 效果 显示 ， 那 么 ， 如 何 实 
现 文字 的 滑动 效果 呢 ? | 
专家 解答 
1， 解决 思路 
文字 的 滑动 效果 可 以 使 用 <marquee>...</marquee> 标 签 实现 。 | 
<marquee>...</marquee> 标 签 是 成 对 出 现 的 ， 首 标签 <marquee> 和 尾 标签 </marquee> 之 间 的 | 
内 容 就 是 滚动 内 容 。<marquee> 标 签 的 属性 包括 : behavior、bgcolor、direction、width、height、 | 
hspace、vspace、loop、scrollamount、scrolldelay 等 ， 它 们 都 是 可 选 的 。 | 
behavior 属性 | 
behavior 属性 的 参数 值 为 altemate、scroll、slide 中 的 一 个 ， 分 别 表示 文字 来 回 滚动 、 单 方 | 
向 循环 滚动 只 滚动 一 次 .需要 注意 的 是 ,如 果 在 <marquee> 标 签 中 同时 出 现 direction 和 behavior | 
属性 ， 那 么 scroll 和 slide 的 滚动 方向 将 依照 direction 属性 中 参数 的 设置 。 例 如 : | 
<marquee behavior="altermnate"> 我 来 回 滚动 <marquee> 
<marquee behavior="scroll"> 我 单方 向 循环 滨 动 <marquee> 
<marquee behavior="scroll" direction="up" height="30"> 我 改 单方 向 向 上 循环 深 动 </marquee> 
<marquee behavior="slide"> 我 只 深 动 一 次 </marquee> 
<marquee behavior="slide" direction="up"> 我 改 向 上 只 滚动 一 次 了 </marquee> 
bgcolor 属性 | 
文字 滚动 范围 的 背景 颜色 的 参数 值 是 十 六 进 制 (形式: #AABBCC 或 #AA5566 等 ) 或 预定 | 
义 的 颜色 名 字 (如 red、yellow、blue 等 )。 例 如 : | 
<marquee behavior 一 "slide" direction="left" bgcolor="red"> 我 的 背景 色 是 红色 的 </marquee> 


回 direction 属性 | 
文字 滚动 的 方向 ， 属 性 的 参数 值 有 down、left、right、up 共 4 个 单一 可 选 值 ， 分 别 代表 滚 | 
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Ge 
| 动 方向 向 下 、 向 左 、 向 右 、 向 上 。 例 如 : 


<marquee direction='"right"> 我 向 右 滚 动 < marquee> 
<marquee direction="down"> 我 向 下 滚动 <marquee> 


贪 站 | loop 属性 

a | 表示 循环 的 次 数 ， 值 是 正 整数 ， 默 认为 无 限 循环 。 
scrollamount 属性 

表示 运动 速度 ， 值 是 正 整数 ， 默 认为 6。 

scrolldelay 属性 

表示 停顿 时 间 ， 值 是 正 整数 ， 默 认为 0， 单 位 是 毫秒 。 

valign 属性 

表示 元 素 的 垂直 对 齐 方式 ， 值 可 以 是 top、middle、bottom， 默 认为 middle。 

align 属性 

表示 元 素 的 水 平 对 齐 方式 ， 值 可 以 是 left、center、right， 默 认为 left。 

height、width 属性 

表示 运动 区 域 的 高 度 和 宽度 ， 值 是 正 整数 〈 单 位 是 像素 ) 或 百分数 ， 默 认 width=100%， 
height 为 标签 内 元 素 的 高 度 。 

hspace、vspace 属性 

表示 元 素 到 区 域 边界 的 水 平 距 离 和 垂直 距离 ， 值 是 
正 整数 ， 单 位 是 像素 。 

2. 应 用 示例 

下 面 使 用 <marquee>...</MARQUEE> 标 签 制作 公告 


信息 滑动 显示 ， 结 果 如 图 3.9 所 示 。 ee 
关键 代码 如 下 ; 图 3.9 最 新 公告 信息 滑动 显示 效果 


<marquee scrollamount="1" scrolldelay="40" direction="up" width="220" height="120" onMouseOver= 
"this.stopO" onMouseOut="this.startO"> 
<?php 
| $board=array('PHP24 堂 课 出 版 了 ! ''PHP 疑难 300 问 即将 出 版 1 , 共 贺 PHP 从 入 门 到 精通 荣获 最 
| 畅销 图 书 奖 ! '); 
| for(G$i=0:$i<count($board):$i+H){ 
| > 
<div class="divList"> &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<a target="_blank" class= 
"huil2"><?php echo $board[$i];?></a></div> 


<2php 
<MARauEE> 
问题 85 ”如何 实现 自动 登录 ? 
| 问题 亲 


所 谓 自动 登录 ， 是 指 用 户 第 一 次 成 功 登 录 网 站 后 ， 在 一 段 时 间 内 再 次 登录 这 个 网 站 时 不 再 
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需要 填写 用 户 名 和 密码 ， 而 是 可 以 直接 进入 。 那 么 这 个 功能 是 如 何 实 现 的 呢 ? 其 原理 又 是 怎样 
的 呢 ? 
专家 解答 

1. 解决 思路 

本 功能 实现 的 关键 是 COOKIE 技术 。 其 原理 是 : 当 用 户 第 一 次 登录 时 , 将 登录 信息 存储 到 
COOKIE 变量 中 ,并且 设置 COOKIE 的 过 期 时 间 ; 然后 当 用 户 再 次 在 本 机 中 登录 此 网 站 时 ,， 判 
断 COOKIE 变量 是 否 存 在 ， 如 果 存在 则 直接 登录 网 站 。 

其 中 , 创建 COOKIE 应 用 的 是 setcookie0 函 数 ， 完 成 COOKIE 变量 的 创建 、 赋 值 以 及 过 期 
时 间 的 设置 ， 获 取 COOKIE 变量 中 的 值 应 用 的 是 $_ COOKIE[] 全 局 变量 。 
由 于 COOKIE 是 HITP 头 标 的 组 成 部 分 , 作为 头 标 必须 在 页 面 其 他 内 容 之 前 发 送 , 也 必须 
最 先 输出 , 所 以 在 setcookie() 函 数 之 前 不 能 有 任何 内 容 输 出 , 即使 是 一 个 HTML 标记 、 一 个 echo 
语句 甚至 一 个 空 行 都 会 导致 程序 出 错 ， 这 就 是 setcookie() 函 数 ， 下 面 是 它 的 语法 : 

bool setcookie(string name[,string value[,int expire[, string path[,string domain[,int secure]]]]]) 
setcookie() 函 数 的 参数 说 明 如 表 3.1 所 示 。 


表 3.1 setcookie() 函 数 的 参数 说 明 


参数 说 了 明 举例 
si 可 以 通过 $ COOKIE['cookiename '] 调 用 变量 名 为 cookiename 
name ”|COOKIE 的 变量 名 GOONIE 
: 上， 该 值 保存 在 
value CR 这 旺 的 信访 信 保存 在 千 可 以 通过 $_COOKIE['values '] 获 取 名 为 values 的 值 


户 端 ， 不 能 用 来 保存 敏感 数据 
。 |5OOKIE 的 过 期 时 间 ，expire 是 标准 | 训 呆 不 设置 COOKIE 的 过 期 时 间 ， 那 么 COOKIE 将 永远 有 
expire “| 的 UNIX 时 间 标 记 ， 可 以 用 time0 函 效 ， 除 非 手动 将 其 出 除 
数 或 mktimeO 函 数 获取 ， 单 位 为 秒 
如 果 该 参数 设置 为 “/”， 则 它 在 整个 domain 内 有 效 ， 如 
path ”|COOKIE 在 服务 器 端的 有 效 路 径 。 | 果 设置 为 “/10.9”,， 则 它 在 domain 下 的 /10.9 目 录 及 子 目 录 
内 有 效 。 默 认 是 当前 目录 
如 果 要 使 COOKIE 在 mrbccd.cn 域 名 下 的 所 有 子 域 都 有 效 ， 
则 应 设置 为 mrbccd.cn 
指明 COOKIE 是 否 仅 通过 安全 的 | 如 果 值 为 1， 则 COOKIE 只 能 在 HITPS 连 接 上 有 效 ， 如 果 
HTTPS， 值 为 0 或 1 值 为 默认 值 0， 则 COOKIE 在 HITP 和 HTTPS 连 接 上 均 有 效 


domain |COOKIE 有 效 的 域名 


secure 


2. 应 用 示例 
本 示例 实现 一 个 自动 登录 的 功能 。 其 具体 步 又 如 下 : 
(1) 创建 index.php 文件 , 创建 表单 ,添加 用 户 名 、 密 码 文本 框 和 保存 时 间 选 择 单 选 按钮 。 
最 关键 的 是 编写 PHP 脚本 ， 通 过 $_ COOKIE 获取 COOKIE 中 存储 的 用 户 名 和 密码 数据 ， 并 将 
其 作为 用 户 名 和 密码 的 默认 值 。 其 关键 代码 如 下 : 
<form id="forml1" name="forml" method="post" action="index_ok.php"> 
<input id="lename" name—"name" value="<?php echo $ COOKIE['name']:?>" type="text" class—"txt"/> 
<input id="lgpwd" name="pwd" value="<?php echo $ COOKIE['pwd]:?>" type="password" class="txt"/> 
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<input type="radio" name="times" value="86400" />1 天 

<input type="image" name="imageField" src="images/dl.gif"' /> 

<input type="image" name="imageField2" onclick="form resetO:retum false:" src="images/cz.gif" /> 
</form> 


(2) 创建 index_ok.php 文件 ， 通 过 $_ POST 方法 获取 表单 中 提交 的 数据 ， 验 证 用 户 输入 的 
用 户 名 和 密码 是 否 正 确 。 如 果 正 确 ， 则 通过 setcookie0 函 数 创建 COOKIE， 存 储 用 户 名 和 密码 ， 
并 根据 表单 提交 的 时 间 设置 COOKIE 过 期 时 间 ， 同 时 跳 转 到 main.php 页 面 ， 如果 不 正确 ， 则 
给 出 提示 信息 ， 并 跳 转 到 index.php 页 面 。 其 关键 代码 如 下 : 


| <2php 
| if(!empty ($_POST [name'] ) and !empty($ POST [pwd])){ /判断 用 户 名 和 密码 是 否 为 空 
| if($_POST [name'] =— "mr" && $ POST [pwd] 一 "mrsoft) { 
! setcookie ( "name", $ POST [mame],time () + $_POST ['times'] ); 
/设置 COOKIE 有 效 时 间 为 1 小 时 
setcookie ( "pwd", $_ POST [pwd], time (0 + $_POST [imes'] ); 
// 设 置 COOKIE 有 效 时 间 为 1 小 时 
echo "<script>alert('succeed!");window.location.href="main.php';</script>"; 
} else{ 
echo "<script>alert('false!');window.location.href='index.php';</script>"; 


| <input name="times" type="radio" value="3600" checked="checked" />1 小 时 


} 
}else{ 
echo "<script>alert(NULL'"):;window.location.href='index.php';</script>"; 
) 


?> 


| (3) 创建 main.php 文件 ， 首 先 根据 $ COOKIE 获取 的 COOKIE 值 判 断 用 户 是 否 具 有 访问 
| 权限 ， 如 果 有 则 可 以 看 到 本 页 内 容 ， 否 则 将 给 出 提示 信息 ， 跳 转 到 index.php 页 面 。 其 关键 代 
| 码 如 下 : 
| <2php 
| 这 $_COOKIE[name] 一 ""){ /根据 COOKIE 的 值 , 判断 浏览 者 是 否 具 有 访问 该 页 面 的 权限 
| echo "<script>alert(' 您 不 具有 访问 该 页 面 的 权限 ! "); window.location.href='index.php';</script>"; 
| }else{ // 如 果 正 确 则 输出 主页 内 容 

> 
| <! 一 省 略 了 HIML 内 容 --> 
| <2php 
| } 
| > 
| 运行 本 示例 ， 如 果 是 第 一 次 登录 ， 则 需要 填写 用 户 名 和 密码 才能 够 登录 ;如果 不 是 第 一 次 
| 登录 ， 则 不 需要 输入 用 户 名 和 密码 ， 因 为 9 COOKIE 会 从 COOKIE 中 读 取 到 用 户 名 和 密码 , 用 
| 户 直接 单 击 登录 按钮 即 可 。 登 录 成 功 后 进入 main.php 页 面 ， 运 行 结果 如 图 3.10 所 示 。 
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! COOF 下 自动 登录 功能 
| 保存 时 间 : 会 1 时 个] 天 


| 图 3.10 进入 网 站 主页 
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专家 点 评 
在 使 用 COOKIE 时 , 如 果 设置 COOKIE 的 过 期 时 间 , 那么 浏览 器 将 会 保存 COOKIE 数据 ，| 

即使 用 户 重新 启动 计算 机 ， 只 要 未 过 期 ，COOKIE 数据 就 一 直 有 效 。 


问题 86 ”如 何 统计 用 户 在 线 时 间 ? 


问题 阐述 | 
所 谓 用 户 在 线 时 间 就 是 当 用 户 登 录 到 页 面 开始 ， 到 关闭 浏览 器 结束 的 时 间 差 ,而 COOKIE | 
的 生命 周期 在 默认 情况 下 关闭 浏览 器 时 自动 删除 。 这 非常 符合 在 线 统计 时 间 的 特点 ， 是 很 多 服 | 
务 类 网 站 优先 选择 的 统计 手段 。 | 
专家 解答 
1. 解决 思路 | 
统计 用 户 在 线 时 间 的 原理 :通过 COOKIE 保存 用 户 登录 到 页 面 的 时 间 惟 和 离开 网 站 时 的 时 | 
间 戳 ， 将 这 两 个 时 间 戳 做 减法 运算 ， 所 得 到 的 就 是 用 户 的 在 线 时 间 。 其 具体 做 法 如 下 : | 
用 户 登 录 时 间 : 
$str = time(); 
setcookie('times', $str); 
用 户 退 出 时 间 : 
$string = time0-$_ COORKIE[Yimes']; 
echo "在 线 时 间 : "date("H:i:s",$string)." 秒 ! "."<br>"; 
例如 ， 通 过 创建 COOKIE， 保 存 用 户 登 录 到 页 面 的 时 
间 惟 ， 在 用 户 单 击 退 出 按钮 时 计算 用 户 在 线 时 间 。 运 行 结 
果 如 图 3.11 所 示 。 


实现 步 又 如 下 : 站 各 的 在 六 时间 
(1) 在 index.php 文件 中 创建 COOKIE， 存 储 用 户 的 
登录 时 间 惟 。 其 关键 代码 如 下 : 图 3.11 统计 用 户 的 在 线 时 间 
<2php 
include_once 'conn/conn php'; // 执 行 连接 数据 库 的 操作 
$str = time(); 
setcookie('times', $str); 


if($_ COOKIE[namel]!="" and $ COOKIE[id]!=""){ | 
$sql = "select * from tb login where name = ".$ COOKIE[name]." and login id = | 
OO | 

Snum = $conne->getRowsNum($sq]l); /返回 查询 结果 

$conne->close rst0: /关闭 数据 库 

if($num != 0 or $num (="){ 

header("Location: default.php"); 
}else{ 
> 
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| (2) 创建 information.php 文件 ， 应 用 系统 的 当前 时 间 戳 减 去 COOKIE 中 存储 的 登录 时 间 
| 戳 ， 即 获取 到 系统 的 在 线 时 间 的 时 间 戳 ， 并 通过 date0) 函 数 对 这 个 时 间 惟 进行 格式 化 ， 并 输出 
| 用 户 的 在 线 时 间 ， 同 时 输出 登录 用 户 的 瑟 和 机 器 名 称 。 其 关键 代码 如 下 : 


会 A 
es | $string = time()-$ COOKIE['times']: 
| echo "在 线 时 间 : "date("H:is",$string)." 秒 ! "."<br>"; 
echo "您 的 他 地址: ".getenv(REMOTE _ADDR")."<br>"; 
| echo "您 的 机 器 名 称 : ".gethostbyaddr(getenv(REMOTE ADDRJ); 
?> 
问题 87 ”如 何 限制 用 户 访问 网 站 的 时 间 ? 
| 问题 阐述 


| 默认 情况 下 ，COOKIE 生命 周期 是 以 关闭 浏览 器 为 基准 的 ， 也 就 是 说 ， 如 果 用 户 不 关闭 浏 
| 览 器 并 且 没有 对 COOKIE 进行 设置 ， 则 COOKIE 永远 也 不 会 过 期 失效 。 假 如 用 户 在 互联 网 发 
| 布 网 站 ， 有 成 百 上 千 的 用 户 浏览 这 个 网 站 ， 并 且 在 线 浏览 用 户 数量 一 直 增 加 。 如 果 不 对 用 户 访 
| 间 网 站 的 时 间 进 行 限制 ， 则 结果 只 能 是 服务 器 资源 耗 尽 ， 网 站 瘫痪 。 那 么 如 何 限制 用 户 访问 网 


| 站 的 时 间 呢 ? 
专家 解答 
1. 解决 思路 
首先 通过 setcookie0 函 数 创建 COOKIE 并 设置 COOKIE 的 有 效 时 间 , 然 后 通过 判断 COOKIE 
变量 是 否 存 在 和 COOKIE 变量 的 值 是 否 为 空 来 控制 访问 网 站 的 时 间 。 其 关键 代码 如 下 : 
<2php 
session start(); // 初 始 化 SESSION 变量 
$session id=session id(); // 获 取 SESSION ID 
setcookie("start",$session_id,time()+10); // 定 义 COOKIE 变量 , 设置 COOKIE 的 有 效 时 间 是 10 秒 
?> 


| 在 上 述 的 方法 中 ， 定 义 COOKIE 的 过 期 时 间 为 10， 即 网 站 的 访问 时 间 为 10 秒 ， 其 原理 就 
| 是 如 此 ， 至 于 在 实际 的 应 用 中 ， 设 置 网 站 的 访问 时 间 为 多 长 ， 可 由 用 户 自行 决定 。 

2. 应 用 示例 

本 示例 通过 设置 COOKIE 限制 用 户 访问 网 站 的 时 间 ， 运 行 结果 如 图 3.12 和 图 3.13 所 示 。 


FANXING =: 


| ran 


您 访问 网 站 的 时 间 到 了 ! 


| 图 3.12 COOKIE 未 失效 ， 访 问 网 站 图 3.13 COOKIE 失效 
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实现 步骤 如 下 : 


(1) 创建 index.php 文件 。 首 先 ， 初 始 化 SESSION 变量 ， 获 取 SESSION ID， 然 后 , 通 ， | 
过 setcookie0 函 数 创建 COOKIE， 并 将 SESSION_ID 作为 COOKIE 值 ， 同 时 设置 COOKIE 的 | 
有 效 时 间 为 10 秒 。 | 

(2) 在 页 面 中 通过 判断 COOKIE 变量 的 值 是 否 为 空 来 限制 用 户 访问 网 站 的 时 间 。 其 关键 | | 
代码 如 下 : 


<?php 
ifisset($_COOKIE['start])l$_COOKIE['start] 一 Ssession id){ 


RE /省 略 了 网 页 中 的 代码 
<?php 

jelse{ 

echo "<hl style ='colorred:> 您 访问 网 站 的 时 间 到 了 <hl>"; 
} 


> 


问题 88 ”如 何 掌 榨 登 录用 户 的 权限 ? 


问题 阐述 | 

在 论坛 中 ， 如 果 是 管理 员 ， 则 可 以 发 布 公告 信息 ; 如 果 是 普通 用 户 ， 则 不 可 以 发 布 公告 信 
息 。 那 么 程序 中 是 如 何 对 登录 用 户 的 权限 进行 判断 的 呢 ? | 
专家 解答 

1. 解决 思路 | 

在 用 户 登录 成 功 后 ,将 登录 用 户 信息 存储 到 一 个 SESSION 变量 中 ,这 个 SESSION 变量 值 | 
在 网 站 中 本 次 会 话 结束 之 前 ) 一 直 有 效 ， 那 么 就 可 以 根据 这 个 SESSION 变量 的 值 判断 用 户 | 
的 权限 。 | 

创建 一 个 SESSION 变量 ,首先 要 通过 session_start0 函 数 初始 化 SESSION 变量 ,然后 通过 | 
预定 义 变 量 $_SESSION[] 直 接 为 变量 赋值 。 其 实现 代码 如 下 : | 

session start():; 

$_SESSION["name"] = "value"; 

通过 预定 义 变量 $_SESSION[] 定 义 SESSION 变量 。“name” 为 设置 变量 的 名 称 ,“value” 
表示 设置 变量 的 值 。 

2. 应 用 示例 | 

通过 SESSION 变量 掌控 登录 用 户 的 权限 。 如 果 是 管理 员 登 录 ， 则 可 以 添加 公告 信息 ;而 | 
如 果 是 普通 用 户 ， 则 只 可 以 浏览 论坛 中 的 帖子 。 其 运行 结果 如 图 3.14 和 图 3.15 所 示 。 | 

上 述 是 以 管理 员 身 份 进行 登录 ， 进 入 main php 主页 后 ， 可 以 单 击 “ 公 告 ” 超 链接 ， 发 布 公 | 
告 信息 ， 如果 是 以 普通 用 户 身份 登录 ， 那 么 在 main php 主页 中 ,“ 公 告 ” 超 链接 是 不 可 以 点 击 | 
的 ， 因 为 普通 用 户 不 具备 发 布 留言 的 权限 。 | 
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图 3.14 管理 员 登 录 图 3.15 管理 员 页 面 


实现 步骤 如 下 : 

(1) 创建 mdex.php 文件 ， 添 加 表单 ， 设 计 用 户 登录 页 面 ， 并 获取 表单 提交 的 用 户 名 和 密 
码 ， 在 本 页 中 对 用 户 提 交 的 数据 进行 判断 。 首 先 ， 初 始 化 SESSION 变量 ， 判 断 用 户 提交 的 值 
是 否 为 空 ， 同 时 判断 如 果 登 录用 户 是 管理 员 ， 则 为 SESSION 变量 type 赋值 为 0， 如果 登录 用 
户 是 普通 用 户 ， 则 为 SESSION 变量 type 赋值 为 1。 其 关键 代码 如 下 : 


<?php 


session_ start(); 


这 $ ] 


} 


?> 


POST[sub'] 一 true){ 
这 $_ POST[user]!=" && $ POST[pwd]!=""){ /判断 用 户 名 和 密码 是 否 为 空 
这 $_ POST[user] 一 "mr' && $_POST['pwd']=="mrsoft"){ /判断 是 否 是 管理 员 


$_SESSION['type] = 0; /为 SESSION 变量 赋值 为 0 
echo "<scriptf>alert(' 管 理 员 登录 成 功 ! ) window.location.href='main.php':</script>"; 
}else{ 


$_SESSION['type']= 1  // 如 果 不 是 管理 员 ， 则 为 SESSION 变量 赋值 为 1 
echo "<script>alert(' 普 通用 户 登 录 成 功 ! ); window.location href=mainphp':</script>"; 


} 

$_SESSION[user] =$_POST['"user]: // 将 登录 用 户 名 存储 到 SESSION 变量 user 中 

$_SESSION['pwd'] =$_POST['pwd']: /将 登录 密码 存储 到 SESSION 变量 pwd 中 
jelse{ 

echo "<script>alert( 用 户 名 和 密码 不 能 为 室 ! ); windowlocation href~'index.php';</script>"; 
bh 


(2) 创建 main.php 文件 ， 编 写 网 站 的 主页 面 。 首 先 ， 初 始 化 SESSION 变量 ， 然 后， 根据 
SESSION 变量 type 的 值 定义 页 面 的 输出 内 容 ， 如 果 SESSION 变量 的 值 为 0， 则 说 明 登 录用 户 
是 管理 员 , 即 具 备 发 布 公告 的 权限 ; 如 果 SESSION 变量 的 值 为 1, 则 说 明 登 录用 户 是 普通 用 户 ， 
即 不 具备 发 布 公告 的 权限 ， 也 就 是 说 控制 发 布 公告 的 超 链 接 不 输出 。 其 关键 代码 如 下 : 


<2php 


Session_startO: /初始化 SESSION 变量 


2 


<2?php 
if($_SESSION['type']="0"){// 根 据 SESSION 变量 type 的 值 决定 是 否 输出 发 布 公告 的 超 链接 


和 
<area S| 


一 "Tect" coords="330,64,385,100" href="mes.php" /> 
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<?php 
> | 
(3) 创建 mesphp 文件 ， 编 写 发 布 公告 信息 的 页 面 ， 根据 SESSION 变量 type 的 值 决定 用 | 
户 是 否 具有 访问 此 页 面 的 权限 。 | 
专家 点 评 


SESSION 生命 周期 从 用 户 访问 页 面 开始 到 断 开 与 网 站 的 链接 为 止 。 用 户 在 每 一 次 链接 时 ， 
PHP 都 会 自动 生成 一 个 唯一 的 SESSION ID 以 标识 当前 用 户 与 其 他 用 户 。 | 


问题 89 如何 让 SESSION 永久 有 效 ? 


问题 阐述 | 
在 网 站 中 常常 会 出 现 这 样 的 情况 ， 在 以 一 个 管理 员 的 身份 登录 网 站 后 ， 在 认真 地 回复 客户 | 
的 问题 时 ， 时 间 过 去 大 概 半 个 小 时 之 后 ， 系 统 提示 管理 员 需 要 重新 登录 。 出 现 此 问题 的 原因 是 | 
SESSION 变量 中 存储 的 用 户 已 经 过 期 ， 那 么 该 如 何 解决 这 个 问题 呢 ? 
专家 解答 
1， 解 决 思路 | 
要 让 PHP 中 的 SESSION 永 不 过 期 ， 解 决 方案 可 以 考虑 两 个 方面 ， 第 一 是 对 配置 文件 | 
(php ini) 进行 修改 ， 此 方法 只 需 修改 配置 文件 ， 而 不 必修 改 程序 ， 非 常理 想 ， 但 若 用 户 不 具 | 
备 修改 配置 文件 的 权限 ,那么 就 只 能 考虑 第 二 种 方案 了 ， 即 通过 对 程序 的 修改 , 实现 SESSION | 
永 不 过 期 。 | 
说 明 : 
这 里 所 指 的 SESSION 永 不 过 期 ， 不 包括 手动 销毁 SESSION. 


2. 通过 修改 配置 文件 实现 SESSION 永 不 过 期 
打开 php.ini 配置 文件 ， 定 位 到 如 下 3 个 位 置 ， 并 对 它们 的 值 进行 修改 : 
session.use cookies=1 
设置 它 的 值 为 1， 通 过 COOKIE 来 传递 SESSION ID。 
session.cookie lifetime =0 | 
它 的 作用 是 设置 SESSION ID 在 客户 端 COOKIE 储存 的 时 间 ， 默 认 值 是 0， 表 示 浏 览 器 关 
闭 后 SESSION ID 就 销毁 。 将 它 的 值 设置 的 大 一 些 也 就 延长 了 SESSION 的 有 效 时 间 。 
session.ge_maxlifetime = 1440 | 
它 是 SESSION 数据 在 服务 器 端 储存 的 时 间 ， 如 果 超过 这 个 时 间 ,那么 SESSION 数据 就 自 | 
动 删除 。 同 样 将 它 的 值 也 设置 的 大 一 些 也 就 延长 了 SESSION 在 服务 器 端的 有 效 时 间 。 | 
在 php.ini 文件 中 ， 完 成 对 上 述 3 个 值 的 修改 后 ，SESSION 就 可 以 实现 永久 存储 了 ,前 提 | 
是 只 要 用 户 设置 的 值 足够 大 。 
5 LC 
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3 在 程序 中 实现 SESSION 永 不 过 期 


| 如 果 用 户 不 具备 修改 php.ini 配置 文件 的 权限 ， 那 么 就 只 能 通过 修改 程序 实现 SESSION 的 
| 永久 存储 了 。 通 过 COOKIE 存储 SESSIONID， 其 具体 方法 如 下 : 


到 <2php 
| session start(); /启动 SESSION 
$_SESSION['cou']: // 注 册 SESSION 变量 Count 


isset($PHPSESSID)?session id($PHPSESSID):$PHPSESSID = session id0; 
/如 果 设 置 了 $PHPSESSID， 就 将 SessionID 赋值 为 9PHPSESSID， 和 否则 生成 SESSION ID 


$_SESSION['cou']++:; // 变 量 cou 加 1 
setcookie('PHPSESSID', $PHPSESSID, time()+36000000); /储存 SESSION ID 到 Cookie 中 
echo $cou; // 显 示 SESSION 变量 cou 的 值 
?> 


问题 90 SESSION 购物 车 


| 问题 讲述 

SESSION 购物 车 ， 顾 名 思 义 ， 其 主要 就 是 应 用 SESSION 变量 来 实现 的 。 那 么 ， 如 何 通过 
| SESSION 变量 来 实现 购物 车 功能 呢 ? 

| 专家 解答 

| 1. 解决 思路 

| 购物 车 的 操作 流程 ， 首 先 ， 登 录 到 网 站 中 浏览 商品 。 然 后 ， 购 买 指定 的 商品 ， 进 入 购物 车 
| 页 面 中 ， 可 以 实现 很 多 的 操作 ， 包 括 : 更 改 商品 数量 、 删 除 商品 、 清 空 购物 车 、 继 续 购 物 等 。 
| 最 后 ， 填 写 收 货 人 信息 ， 生 成 订单 。 其 操作 流程 如 图 3.16 所 示 。 


Fo 
中 商品 者 
pr 
中 指定 商品 
1 浏 EE 
芒 小 | 于 一 | 购物 和 EE 
用户 查看 购 槐 车 内 商品 个 组 区 购物 
ET 
商品 全 生计 


图 3.16 购物 车 操作 流程 


| 这 里 所 围绕 的 购物 车 ， 其 实 就 是 在 用 户 登录 网 站 时 为 其 创建 的 两 个 SESSION 变量 ， 其 中 
| 一 个 存储 商品 d， 并 且 以 @ 符 号 分 隔 不 同 id; 另 一 个 对 应 存储 商品 的 购买 数量 ， 同 样 以 @ 符 号 
| 进行 分 隔 。 
| 而 针对 购物 车 进行 的 各 种 操作 ， 也 都 是 围绕 这 两 个 SESSION 变量 中 的 值 进行 的 。 添 加 商 
| 品 ， 就 是 向 SESSION 变量 中 添加 新 的 id 和 对 应 的 数量 值 ， 更 新 商品 购买 数量 ， 则 是 根据 商品 
| 的 记 ， 更 新 对 应 商品 的 数量 ， 删 除 商品 ， 同 样 是 根据 商品 的 DD， 删 除 对 应 商品 数量 ， 清 空 购 
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物 车 ， 则 是 直接 为 这 两 个 SESSION 变量 赋 空 值 。 这 就 是 购物 车 的 基本 原理 ， 至 于 将 其 付 诸 实 | 
践 ， 相 对 会 复杂 一 些 ， 会 涉及 字符 串 与 数组 之 间 的 转换 ， 以 及 数组 中 指定 元 素 值 的 更 新 和 删 | 
除 等 。 

2， 应 用 示例 

购物 车 是 在 网 上 购物 时 使 用 的 一 个 临时 存储 商品 的 “车 辆 "在 本 示例 中 ， 我 们 将 介绍 
SESSION 购物 车 的 实现 方法 ， 其 运行 结果 如 图 3.17 所 示 。 | 

应 用 SESSION 技术 开发 购物 车 的 具体 步 又 如 下 : | 

(1) 创建 index php 页 面 ， 实 现 商品 展示 功能 。 设 计 原理 是 : 从 数据 库 中 读 取 商品 信息 ， | 
将 商品 信息 在 页 面 中 进行 分 栏 、 分 页 显示 ， 并 且 为 商品 设置 购买 和 查看 购物 车 的 超 链接 ,运行 | 
结果 如 图 3.18 所 示 。 | 


r= 


ta zm | 320 | 。 下 县 朋 队 该 基 本 汀 电视 | 
mM | :0 | 记 | ?oo | 了 光量 Bi 大 们 格 :39 询 格 3 35 i | 
EBs Bil Bi | 

| 区 清和 M6 车 > 再 品 侈 富 总 汗 : 9800 元 [| 商品 + 个 部 页 3 不 萄 ;页 藉 3 页 首页 上 一 页 工 -页 悍 责 | 


图 3.17 SESSION 购物 车 图 3.18 ”购物 车 商品 展示 页 面 


(2) 创建 by_commodity.php 文件 ， 实 现 购买 功能 ， 即 向 购物 车 中 添加 商品 。 首 先 创建 一 
个 购物 车 ， 然 后 判断 购物 车 中 是 否 为 空 ， 如 果 为 空 ， 则 将 商品 展示 页 中 “购买 ” 超 链 接 传 递 的 
商品 ID 〈$_GET[id]) 和 数量 添加 到 购物 车 中 ， 如 果 不 为 空 ， 则 判断 添加 商品 的 ID 是 否 在 购物 | 
车 中 已 经 存在 ， 如 果 存 在 则 不 能 重复 添加 ， 否 则 将 商品 ID 添加 到 购物 车 中 ， 最 后 跳 转 到 | 
shopping.php 购物 车 页 面 ， 其 代码 如 下 : | 


<?php | 
session start(); // 初 始 化 SESSION 变量 
header ( "Content-type: text/html: charset=UTF-8" ): /设置 文件 编码 格式 
if($_SESSION["goodsid"]=—"" && $ SESSION["goodsnum"] 一 "1 判断 SESSION 变量 是 否 为 空 
$_SESSION["goodsid"]=$_GET["id"]."@"; // 如 果 SESSION 变量 为 定 ， 则 为 其 赋值 
为 商品 的 ID， 并 以 @ 分 隔 
$_SESSION["goodsnum"]="1@"; /如 果 SESSION 变量 为 空 , 则 为 其 赋值 为 1， 并 以 @ 分 隔 
}else{ // 如 果 SESSION 变量 不 为 空 
S$array=explode("(@",$_SESSION["goodsid"]); / 则 以 @ 为 分 隔 符 ， 将 SESSION 变量 中 
的 数据 写 入 数组 中 
ifin_array($_GET["id"],$aray)){ /判断 数组 中 是 否 存在 指定 的 了 D 
echo "<script>alert(' 该 商品 已 经 被 放 入 购物 车 ! ):history backO:</script>"; 


exit; 


} 
/如 果 数 组 中 不 存在 指定 的 ID， 则 说 明 该 商品 还 未 被 放 入 购物 车 中 


$_ SESSION["goodsid"].=$_GET["id"]."@": /将 该 商品 添加 到 购物 车 中 
$_ SESSION["goodsnum"] ="1@": // 更 改 商 品 数量 

} 

echo "<script>window.location.href='shopping_car.php';</script>"; 

> 


“rs 
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六 

(3) 创建 shopping php 购物 车 页 面 ， 对 购买 商品 进行 统计 ， 包 括 购买 数量 、 每 种 商品 的 
金额 和 总 金额 ,并且 实现 删除 购物 车 中 商品 、 更 改 购物 车 中 商品 数量 和 清空 购物 车 的 功能 ， 同 
时 设置 “继续 购买 ”的 超 链 接 ， 其 代码 可 参考 本 书 配套 资源 。 

(4) 创建 change_commodity_counts.php 文件 ， 实 现 更 改 购物 车 中 商品 的 数量 。 首 先 获取 
shopping.php 页 面 表单 中 提交 的 商品 数量 和 商品 ID,， 然后 通过 正则 表达 式 验证 商品 数量 是 否 为 
正 整数 ， 最 后 根据 商品 ID 找到 指定 的 商品 ， 更 改 商 品 数量 ， 并 返回 购物 车 页 面 ， 其 代码 如 下 : 


<2php 
session_ start(); 
header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
$id=$_POST["id"]; // 获 取 商 品 人 DD 
$num=$_POST["goodsnum"]; // 获 取 商 品 数 量 
Spreg="/^[0-9]*[0-9]$I^[0-9]*[0-9]8/"; // 编 写 正则 表达 式 
if($num—"){ /判断 提交 的 值 是 否 为 空 
echo "<script>alert( 数 量 不 能 为 空 !):history.backO:</script>"; 
exit; 
}else if(!preg_match($preg,$num,$str){ 1/ 判断 提交 的 数据 是 否 是 正 整数 
echo "<script>alert(' 数 量 只 能 为 正 整 数 1");history.back();</script>"; 
exit; 
} 


S$arrayid=explode("@",$_SESSION["goodsid"]); 
$arraynum=explode("@",$_ SESSION["goodsnum'"]); 


$key=array_search($id,$arrayid): /在 数组 中 搜索 给 定 的 值 ， 如 果 成 
功 则 返回 相应 的 键 名 

$arraynum[$key]=$num:; // 更 改 商 品 数量 

$_SESSION["goodsnum"]=implode("(@",$arraynum); // 更 改 商 品 数 量 

echo "<script>window.location.href='shopping_car.php';</script>"; 

?> 


(5) 创建 delete_commodity.php 文件 ， 删 除 购物 车 中 指定 的 商品 。 首 先 获取 超 链 接 中 传递 
的 商品 ID， 然 后 将 SESSION 中 的 字符 串 数据 转换 成 数组 ， 并 根据 商品 ID 获取 指定 要 删除 的 
商品 的 键 名 ， 最 后 根据 键 名 为 指定 的 数组 赋 空 值 ， 并 重新 将 数组 添加 到 购物 车 中 。 

(6) 创建 clear_ shopping_carphp 文件 ， 清 空 购物 车 ， 即 将 SESSION 变量 $ SESSION 
["goodsid"] 和 $_SESSION["goodsnum"] 赋 值 为 空 。 其 代码 如 下 : 


<2?php 
session _startO: /初始化 SESSION 变量 
header ( "Content-type: text/html: charset=UTF-8" ): /设置 文件 编码 格式 
$_SESSION["goodsid"]=""; /为 购物 车 的 ID 赋值 为 空 
$_ SESSION["goodsnum"]="": /为 购物 车 数量 赋值 为 空 
echo "<script>window.location.href='shopping_car.php':</script>"; // 完 成 清空 操作 , 跳 转 到 购物 页 面 
?> 
专家 点 评 


在 应 用 SESSION 变量 实现 购物 车 功能 时 ， 最 容易 出 现 的 一 个 错误 就 是 忘记 初始 化 
SESSION 变量 , 如 果 在 文件 中 没有 初始 化 SESSION 变量 ,那么 在 页 面 之 间 就 获取 不 到 SESSION 
变量 中 存储 的 数据 。 

清空 购物 车 中 的 商品 可 以 采用 直接 为 SESSION 变量 赋 空 值 的 方法 ， 也 可 以 直接 应 用 
session destroy() 函 数 删 除 SESSION， 或 者 应 用 unset0 函 数 删除 指定 的 SESSION 变量 。 
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问题 91 headerO 函 数 都 有 哪些 功能 ? 


问题 阐述 | 用] 

通过 header() 函 数 可 以 发 送 一 个 原始 的 HTTP 头 ,那么 通过 它 具 体能 实现 什么 样 的 功能 呢 ? 
专家 解答 

1. HTIP 头 | 

服务 器 发 送 回来 的 HITP 响应 包含 以 下 信息 : 用 于 识别 响应 主体 内 容 的 头 (header)， 发 送 | 
响应 的 服务 器 ， 响 应 消息 有 多 少 字 节 ， 以 及 响应 何 时 发 出 等 。PHP 和 Apache 已 经 完成 对 头 信 | 
息 的 处 理 : 将 文档 识别 为 HTML 和 计算 HTML 页 面 的 长 度 等 。 绝 大 多 数 Web 程序 不 需要 自己 | 
设置 头 。 但 是 ， 如 果 想 让 服务 器 返回 的 不 是 HIML， 或 者 想 设置 页 面 的 过 期 时 间 ， 重 定向 客户 | 
端 浏览 器 到 另 一 个 地 址 ， 或 是 产生 一 个 特定 的 HITP 错误， 就 需要 使 用 header() 函 数 。 

2.，header0 浮 数 

header0 函 数 的 语法 如 下 : 

void header ( string string [, bool replace [, int http_response_code]] ) 
header() 函 数 的 参数 说 明 如 表 3.2 所 示 。 


表 3.2 header() 函 数 的 参数 说 明 


参数 说 了 明 
strin 必要 参数 。 输 入 的 头 部 信息 
ee 可 选 参数 。 指 明 是 蔡 换 掉 前 一 条 类 似 的 标 头 还 是 增加 一 条 相同 类 型 的 标 头 。 默 认 


为 替换 ， 但 如 果 将 其 设 为 false， 则 可 以 强制 发 送 多 个 同类 标 头 
可 选 参数 。 强 制 将 HTTP 响 应 代码 设 为 指定 值 ， 此 参数 是 php5.3.0 以 后 添加 的 


http_response code 


设置 header 一 定 要 在 生成 主体 内 容 之 前 完成 ， 这 意味 着 所 有 header() (或 setcookie0， 如 
果 你 想 设置 COOKIE) 要 在 文件 的 最 前 面 ， 在 <html> 标 签 之 前 。 

3. 设置 文档 类 型 

Content-Type 头 指定 被 返回 文档 的 类 型 。 它 通常 是 “text/html”, 指明 它 是 一 个 HTML 文档 ， | 
ee 些 有 用 的 文档 类 型 ， 例 如 ,“textplain” | 

。 这 个 类 型 就 类 似 于 自动 的 “查看 源 代码 ”， 它 在 调试 时 很 有 用 。 

例如 ， 应 用 header() 函 数 ， 设 置 文档 类 型 为 image/png， 其 代码 如 下 : 

header("Content-type:image/png"); /发 送 头 部 信息 ， 生 成 png 的 图 片 文件 
4 重 定 向 
通过 header0 函 数 可 以 向 浏览 器 发 送 一 个 新 的 URL， 并 让 浏览 器 转向 到 该 地 址 。 这 样 的 


定向 〈redirection) 操作 只 需要 通过 设置 Location 头 即 可 。 例 如 ， 通 过 header0 两 数 重 定向 到 | 
http://www.mrbccd.com， 其 代码 如 下 : 
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header("Location: http://www.mrbccd.com"); 
重 定 向 操作 更 倾向 于 绝对 路 径 ， 如 果 提 供 相对 的 URL (如 “/index.php”)， 则 重 定向 会 在 
服务 器 内 部 进行 。 这 种 方法 很 少 用 ， 因 为 浏览 器 并 不 知道 它 得 到 的 页 面 是 否 是 所 请 求 的 。 如 果 
在 新 的 文档 中 存在 相对 URL, 则 浏览 器 会 将 它们 解释 成 相对 于 所 请 求 的 文档 , 而 不 是 被 发 送 的 
文档 。 

5. 设置 过 期 时 间 

服务 器 可 以 显 式 地 通知 浏览 器 (或 者 那些 存在 于 服务 器 的 浏览 器 之 间 的 代理 服务 器 缓存 ) 


| 文档 的 过 期 时 间 。 代 理 服务 器 和 浏览 器 缓存 在 过 期 之 前 可 保持 文件 ， 或 提前 结束 它 。 重 新 载 入 


| 一 个 被 缓存 的 页 面 不 需要 和 服务 器 进行 通信 。 但 是 尝试 获取 一 个 已 经 过 期 的 文档 就 需要 与 服务 
| 器 取得 联系 。 


| 期 字符 中 


为 一 个 文档 设置 过 期 时 间 ， 应 用 的 是 Expires 头 : 
header("Expires: Mon, 08 Jul 2011 08:08:08 GMT"); 
例如 ， 控 制 文档 在 页 面 生成 后 的 两 小 时 后 过 期 。 使 用 tme0 和 gmstrfime() 函 数 生成 过 期 日 


<?php 

$now=time0; // 获 取 系 统 当前 时 间 戳 
S$then=gmstrftime("%a,%d %b %Y %H:%M:%s GMT",$now + 60*60*2); /格式 化 时 间 
header("Expires:$then"); // 定 义 文档 过 期 时 间 


2> 
6. HTTP 认证 
HTTP 认证 (HTTP authentication) 通过 请 求 的 header 和 响应 状态 来 工作 。 浏 览 器 可 以 将 用 


户 名 和 密码 放 在 请 求 的 头 中 发 送 。 如 果 认 证 凭证 〈credential， 即 指 用 户 名 和 密码 ) 未 发 送 或 不 
| 匹配 ， 服 务 器 将 发 送 一 个 “401 Unauthorized” 响 应 并 通过 WWW 认证 头 来 确定 当前 认证 的 区 
| 域 (realm) (一 个 字符 串 ， 诸 如 “Mary's Pictures” 或 “Your Shopping Cart”)。 这 通常 会 导致 济 


览 器 弹出 一 个 “Enter usemame and password for...” 对 话 框 ， 且 该 页 面 会 重新 请 求 更 新 头 中 的 
认证 凭证 。 
为 了 用 PHP 来 处 理 认 证 ， 可 检查 用 户 名 和 密码 〈$_SERVER 数组 中 的 PHP_AUTE _USER 
和 PHP AUTH PW 两 个 元 素 ) 并 调用 header0 函 数 来 设置 区 域 ， 然 后 发 送 一 个 “401 
Unauthorized” 响 应 。 其 关键 代码 如 下 : 
header(CWWW-Authenticate:Basic realm="Top Secret Files"); 
header("HTTP/1.0 401 Unauthorized"): 
另外 ，header() 函 数 还 可 以 强制 客户 端 每 次 访问 页 面 时 获取 最 新 资料 ， 而 不 是 使 用 存在 于 
客户 端的 缓存 。 其 关键 代码 如 下 : 
header("Expires: Mon, 08 Jul 2011 08:08:08 GMT"):; // 设 置 页 面 的 过 期 时 间 ( 用 格林 威 
治 时 间 表 示 ) 
header("Last-Modified:".gmdate("D, d MY H:i:s")."GMT"): ”// 设 置 最 后 更 新 日 期 (用 格林 威 治 
时 间 表示 ), 使 浏览 器 获取 最 新 资料 
header("Cache-Control: no-store,no-cache, must-revalidate"): 。 // 控 制 页 面 不 使 用 缓存 
header("Cache-Control: post-check=0,pre-check=0" false); // 控 制 页 面 不 使 用 缓存 
header("Pragma: no-cache"): // 参 数 (与 以 前 的 服务 器 兼容 )， 即 兼容 HTTP1.0 协议 
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7. 文件 下 载 

根据 服务 器 返回 的 头 信息 ， 应 用 header 函数 可 以 完成 文件 的 下 载 操作 。 其 关键 代码 如 下 : 
header("Content-type:application/octet-stream"); // 输 出 MIME 类 型 
header("Accept-ranges:bytes"); // 接 受 的 范围 单位 
header("Accept-length:".filesize($path)); // 文 件 长 度 


header("Content-Disposition:attachment;filename=".$filename); // 默 认 时 文件 保存 对 话 框 中 的 文件 | 
名 称 


例如 ， 通 过 header() 函 数 生 成 Excel 格式 的 下 载 文 件 ， 其 关键 代码 如 下 : 


<?php 
header("Content-type:application/vnd.ms=excel"); /设置 HITP 标 头 
header("ContentDisposition:filename= 会 议 报 表 .xls"); /定义 下 载 文件 名 称 


?> 
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数据 库 操 作 


于 至 吾 台 于 于 于 于 于 于 于 于 于 于 于 于 于 总 至 台 


问题 92 “MySQL 数据 库 开 发 常识 有 哪些 ? 
问题 93 MySQL 的 常用 命令 有 哪些 ? 
问题 94 MySQL 数据 存储 引 营 InnoDB 
和 MyISAM 的 优势 及 区 别 有 哪 些 ? 

问题 95 ”如 何在 MySQL 中 实现 多 关键 
字 查 询 ? 

问题 96 ”如 何在 MySQL 中 修改 用 户 密 
码 和 创建 权限 用 户 ? 

问题 97 ”在 MySQL 中 两 种 索引 Hash 与 
B-Tree 的 区 别 是 什么 ? 

问题 98 ”应 用 PHP 操作 MySQL 数据 库 
时 应 该 注意 哪些 问题 ? 

问题 99 ”应 用 PHP 向 MySQL 插入 数据 
时 ， 如 何 避 免 产 生 乱 码 ? 

问题 100 MySQL 常见 错误 代码 有 哪些 ? 
问题 101 在 MySQL 中 ， 如 何 实现 多 表 
查询 ? 
问题 102 
接 查询 ? 
问题 103 
接 查询 ? 
问题 104 
查询 ? 
问题 105 
问题 106 
机 热 备份 ? 
问题 107 解决 MySQL 语句 错误 的 最 佳 
方案 是 什么 ? 

问题 108 如 何 执行 数据 的 批量 添加 ? 
问题 109 如 何 规范 地 定义 数据 表 中 的 字 
段 名 称 ? 

问题 110 如 何 查询 指定 时 间 段 的 数据 ? 
问题 111 如 何 按 月 查询 统计 数据 ? 


在 MySQL 中 ， 如 何 实现 内 连 
在 MySQL 中 ， 如 何 实现 外 连 
在 MySQL 中 ， 如 何 实现 赃 套 


如 何 连接 远程 MySQL 服务 器 ? 
如 何 实现 MySQL 数据 库 的 双 


于 于 理 于 于 于 对 和 圣 革 时 竺 竺 于 于 于 时 于 于 至 
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问题 112 如 何 实现 在 删除 数据 前 给 予 提 
示人 信息? 

问题 113 如 何 查询 日 期 型 数据 ? 

问题 114 如 何 使 用 左 连接 (left outer 
join) 查询 数据 ? 

问题 115 如 何 使 用 右 连接 (Tight outer 
join) 查询 数据 ? 

问题 116 如 何 对 统计 结果 进行 排序 ? 
问题 117 如 何 使 用 聚集 函数 SUM 对 学 
生成 绩 进行 汇总 ? 

问题 118 如 何 使 用 聚集 函数 MIN 求 利 
润 最 少 的 商品 ? 

问题 119 如 何 使 用 聚集 函数 MAX 求 销 
售 利润 最 高 的 商品 ? 

问题 120 如 何 使 用 INSERT(sl,x,len,s2) 
芭 数 蔡 换 指定 字符 ? 

问题 121 如 何 使 用 UPPER(s),UCASE(s) 
函数 将 字母 变 成 大 写 ? 

问题 122 如 何 利 用 CSS 样式 实现 分 页 
打印 ? 

问题 123 如 何 使 用 RIRIM(s) 函 数 去 掉 
字符 串 结尾 处 的 空格 ? 

问题 124 MySQL 数据 库 的 备份 和 恢复 
问题 125 ”如 何在 MySQL 中 创建 存储 过 程 ? 
问题 126 如 何 将 Access 数据 导入 SQL 
Server 数据 库 中 ? 

问题 127 如 何 解 决 PHP 不 支持 SQL 
Server 的 问题 ? 

问题 128 如 何 连 接 带 密码 的 Access 数 
据 库 ? 

问题 129 MySQL 和 Server 2000 数据 库 
中 数据 类 型 的 转换 

问题 130 ”PHP 如 何 连接 SQL Server 数据 库 ? 
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问题 92 MySQL 数据 库 开发 常识 有 哪些 ? 


问题 阐述 
MySQL 数据 库 是 目前 程序 员 应 用 较 多 的 数据 库 之 一 ， 选 择 其 作为 数据 库 应 掌握 哪些 基本 


常识 ? 


专家 解答 


MySQL 数据 库 是 目前 比较 流行 的 数据 库 之 一 ,作为 程序 员 应 该 掌握 与 其 相关 的 一 些 常识 ， 


如 存储 引擎 的 选择 、 索 引 的 设计 和 使 用 以 及 大 批量 插入 时 SQL 语句 的 优化 等 ， 可 以 帮助 大 家 | 


更 好 地 使 用 MySQL 数据 库 。 这 里 主要 强调 数据 存储 引擎 的 选择 以 及 索引 的 设计 和 使 用 。 
1 存储 引擎 的 选择 


(1) MyISAM 不 支持 事务 ， 且 不 支持 外 键 ， 优 点 是 访问 速度 高 ， 批 量 插入 速度 快 。 假 设 
大 量 的 操作 由 select、insert 组 成 ， 建 议 采用 该 存储 引擎 。 但 是 在 实际 应 用 中 ， 可 能 出 现 由 于 批 | 


量 插入 过 于 频繁 而 导致 数据 量 到 达 一 定 级 别 ， 从 而 出 现 表 损 坏 的 情况 。 


(2) InnoDB 支持 事务 处 理 ， 但 是 相对 于 前 者 其 处 理 效率 稍 低 ， 并 且 其 索引 及 数据 也 更 占 
用 磁盘 空间 。 在 存储 一 些 关键 数据 并 需要 对 其 进行 事务 操作 时 ， 可 以 选择 InnoDB。 其 前 提 是 ， 


没有 太 大 的 访问 量 。 
2. 索引 的 设计 和 使 用 


没有 索引 的 表 会 在 查询 过 程 中 为 用 户 带 来 不 必要 的 麻烦 ， 除 非 数 据 表 中 数据 量 小 。 下 面 简 


要 分 析 索 引 的 设计 和 使 用 。 


(1) 索引 通常 是 设置 where 子 句 中 的 列 ， 如 果 用 户 设置 select 后 的 列 ， 这 是 没有 任何 意义 | 


的 。 当 然 ， 如 果 用 户 需 要 对 某 列 进行 排序 ，order by 后 的 列 也 是 可 以 建成 索引 的 。 


(2) 使 用 唯一 索引 ， 主 键 就 是 最 好 的 例子 ， 假 设 索引 列 大 都 是 重复 的 ， 如 性 别 ， 这 样 的 


索引 并 不 会 加 快 搜索 速度 。 


(3) 只 要 有 可 能 ， 就 要 尽量 限定 索引 的 长 度 ， 例 如 ， 索 引 列 为 char(100)， 在 其 前 10 个 字 | 
符 大 部 分 都 是 唯一 的 ， 应 设置 索引 的 长 度 为 0， 使 用 短 索 引 可 以 加 快 查询 速度 ， 并 节省 硬盘 | 


空间 。 


还 是 分 别 建 多 个 索引 好 呢 ? 显然 前 者 更 好 , 利用 左前 级 特性 , 只 要 联合 索引 的 最 左 的 列 被 用 到 ， 
那么 索引 都 将 被 使 用 。 

(5) 最 后 需要 声明 的 是 ， 不 要 过 度 使 用 索引 ， 索 引 越 多 ， 插 入 的 速度 就 越 慢 ， 尤 其 是 在 
数据 量 庞 大 时 ， 同 时 ， 大 量 的 索引 也 将 耗费 很 大 的 硬盘 空间 ， 造 成 不 必要 的 浪费 。 
专家 点 评 

存储 引擎 的 选择 与 索引 的 设计 及 使 用 只 是 数据 库 开 发 常识 中 的 一 部 分 。 为 了 使 数据 库 系 统 
能 够 发 挥 其 最 大 功效 ， 用 户 应 选择 最 适合 的 存储 引擎 ， 并 通过 索引 的 设计 来 优化 数据 表 的 查询 
和 提高 整个 数据 库 系 统 的 性 能 。 
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(4) 索引 的 左前 绥 特 性 ， 联 合 索引 实质 上 也 是 建立 了 多 个 索引 ， 那 么 是 建立 联合 索引 好 


加 (本 > 六 须 知道 的 300 个 辣 是 


问题 93 MySQL 的 常用 命令 有 哪些 ? 


全 | 问题 阐述 
在 MySQL 中 ， 有 哪些 常用 的 命令 是 用 户 必 须 了 解 和 掌握 的 ? 


| 专家 解答 
对 于 常用 的 MySQL 命令 ， 无 非 是 “ 增 、 删 、 改 、 查 ”操作 ， 但 是 ， 在 不 依靠 MySQL 第 
| 三 方 可 视 管理 软件 的 情况 下 ， 只 掌握 MySQL 的 查询 和 查看 命令 是 远 远 不 够 的 。 所 以 ， 用 户 还 
| 必须 掌握 其 他 常用 的 命令 。 
| 1 使 用 SHOW 语句 找 在 服务 器 上 当前 存在 的 数据 库 
SHOW DATABASES: 
2. 创建 一 个 数据 库 MYSQLDATA 
CREATE DATABASE MYSQLDATA:; 
3. 选择 你 所 创建 的 数据 库 
mysql> USE MYSQLDATA; 
4. 查看 当前 数据 库 表 
SHOW TABLES: 
5.， 创建 数据 表 
CREATE TABLE MYTABLE (name VARCHAR(20), sex CHAR(])): 
6. 查看 数据 表 结构 
DESCRIBE MYTABLE: 
7.， 向 数据 表 中 添加 记录 
insert into mingri values (“mr”,”mrsoft”); 
8.， 以 文本 形式 将 数据 装 入 数据 表 中 
LOAD DATA LOCAL INFILE “D:/mysql.txt” INTO TABLE MYTABLE:; 


9.， 导入 .sql 文件 命令 


use database; 
source d:/mysql.sql: 


10， 删 除数 据 表 

drop TABLE MYTABLE: 
11. 清空 数据 表 

delete ffom MYTABLE: 
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12. 更 新 数据 表 
update MYTABLE set sid="3" where name= mrsof: 


专家 点 评 


熟练 掌握 常用 的 MySQL 命令 就 如 同一 名 优秀 的 士兵 必须 熟练 掌握 手中 武器 的 使 用 方法 一 
样 。 在 MySQL 日 常 操作 中 ， 熟 练 掌握 这 些 常 用 的 操作 命令 是 十 分 必要 的 ， 可 以 更 好 地 帮助 用 
户 使 用 和 学 习 MySQL 数据 库 。 同 样 ， 作 为 一 名 数据 库 管理 人 员 或 程序 开发 人 员 ， 熟 练 操 作 | 
MySQL 数据 库 更 是 一 项 基本 的 技能 。 | 


问题 94 MySQL 数据 存储 引 沿 InnoDB 和 
MyISAM 的 优势 及 区 别 有 哪 些 ? 


问题 阐述 


常用 的 MySQL 数据 存储 引擎 有 MyISAM (默认 ) 和 InnoDB 等 类 型 ， 这 两 种 类 型 的 存储 | 
引擎 有 什么 区 别 ? 其 各 自 优势 又 是 什么 ? | 
专家 解答 

1. MyISAM 存储 引擎 | 

MyISAM 存储 引擎 是 MySQL 中 常见 的 存储 引擎 , 曾 是 MySQL 的 默认 存储 引擎 .MyISAM | 
存储 引擎 是 基于 ISAM 存储 引擎 发 展 起 来 的 ， 它 解决 了 ISAM 的 很 多 不 足 ， 同 时 它 还 增加 了 很 | 
多 有 用 的 扩展 。 | 
MyISAM 存储 引擎 的 表 存储 成 3 个 文件 ,文件 的 名 字 与 表 名 相同 , 其 扩展 名 包括 frm、MYD | 
和 MYI。 | 

回 “frm: 存储 表 的 结构 。 

回 MYD: 存储 数据 ， 是 MYData 的 缩写 。 

回 MYI: 存储 索引 ， 是 MYIndex 的 缩写 。 

基于 MyISAM 存储 引擎 的 表 支 持 3 种 不 同 的 存储 格式 ， 包 括 静 态 型 、 动 态 型 和 压缩 型 。 

(1) MyISAM 静态 型 

如 果 所 有 表 列 的 大 小 都 是 静态 的 〈 即 不 使 用 xBLOB、xTEXT 或 VARCHAR 数据 类 型 )， | 
则 MySQL 就 会 自动 使 用 静态 MyISAM 格式 。 使 用 这 种 类 型 的 表 性 能 非常 高 ， 因 为 在 维护 和 访 | 
问 以 预定 义 格式 存储 的 数据 时 需要 很 低 的 开销 ， 但 是 ， 这 项 优点 要 以 空间 为 代价 ， 因 为 每 列 都 | 
需要 分 配给 该 列 的 最 大 空间 ， 而 无 论 该 空间 是 否 真 正 地 使 用 。 | 

(2) MyISAM 动态 型 | 

如 果 有 表 列 〈 即 使 只 有 一 列 ) 定义 为 动态 的 使 用 xBLOB、xTEXT 或 VARCHAR), 则 | 
MySQL 就 会 自动 使 用 动态 格式 。 虽然 MyISAM 动态 表 占 用 的 空间 比 静 态 格式 所 占 空间 少 , 但 | 
空间 的 节省 带 来 了 性 能 的 下 降 。 如 果 某 个 字段 的 内 容 发 生 改 变 ， 则 其 位 置 很 可 能 就 需要 移动 ， | 
这 会 导致 碎片 的 产生 。 随 着 数据 集中 的 碎片 增加 ， 数 据 访 问 性 能 就 会 相应 降低 。 这 个 问题 有 两 | 
种 修复 方法 : | 

尽 可 能 使 用 静态 数据 类 型 。 
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| 发 性 。 
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经 常 使 用 OPTIMIZE TABLE 语句 ， 它 会 整理 表 的 碎片 ， 同 时 恢复 由 于 表 更 新 和 删除 
而 导致 的 空间 丢失 。 
(3) MyISAM 压缩 型 
有 时 会 创建 在 整个 应 用 程序 生命 周期 中 都 只 读 的 表 。 如 果 是 这 种 情况 ， 就 可 以 使 用 
myisampack 工具 将 其 转换 为 MYISAM 压缩 表 来 减少 空间 。 在 给 定 硬 件 配置 下 《〈 例 如， 快速 的 
处 理 器 和 低速 的 硬盘 驱动 器 )， 人 性 能 的 提升 将 相当 显著 。 
MyISAM 存储 引擎 的 优势 在 于 占用 空间 小 ， 处 理 速 度 快 。 缺点 是 不 支持 事务 的 完整 性 和 并 


InnoDB 已 经 开发 了 十 余年 ， 遵 循 CNU 通用 公开 许可 (GPL) 发 行 。InnoDB 已 经 被 一 些 


| 重量 级 因特网 公司 所 采用 ， 如 雅虎 、Slashdot 和 Google 等 ， 为 用 户 操作 非常 大 的 数据 库 提供 了 
| 一 个 强大 的 解决 方案 。 InnoDB 给 MySQL 的 表 提 供 了 事务 、 回 深 、 崩溃 修复 能 力 和 多 版 本 并 发 
| 控制 的 事务 安全 。 在 MySQL 从 3.23.34a 开始 包含 InnoDB 存储 引擎 。InnoDB 是 MySQL 上 第 
| 一 个 提供 外 键 约束 的 表 引擎 ， 而 且 InnoDB 对 事务 处 理 的 能 力也 是 MySQL 其 他 存储 引擎 所 无 
| 法 与 之 比拟 的 。 下 面 介 绍 InnoDB 存储 引擎 的 特点 及 其 优 缺点 。 


2.， InnoDB 存储 引擎 
InnoDB 存储 引擎 中 支持 自动 增长 列 AUTO_INCREMENT。 自 动 增长 列 的 值 不 能 为 空 ， 且 


值 必须 唯一 。MySQL 中 规定 自 增 列 必须 为 主键 。 在 插入 值 时 ， 如 果 自 动 增长 列 不 输入 值 ， 则 
| 插入 的 值 为 自动 增长 后 的 值 ， 如 果 输入 的 值 为 0 或 空 (NULL)， 则 插入 的 值 也 为 自动 增长 后 的 
| 值 ， 如 果 插 入 某 个 确定 的 值 ， 且 该 值 在 前 面 没 有 出 现 过 ， 则 可 以 直接 插入 。 


InnoDB 存储 引擎 中 支持 外 键 (FOREIGN KEY)。 外 键 所 在 的 表 为 子 表 ， 外 键 所 依赖 的 表 


为 父 表 。 父 表 中 被 子 表 外 键 关 联 的 字段 必须 为 主键 。 当 删除 、 更 新 父 表 的 某 条 信息 时 ， 子 表 也 
| 必须 有 相应 的 改变 。InnoDB 存储 引 敬 中， 创建 的 表 的 表 结构 存储 在 .frm 文件 中 。 数 据 和 索引 
| 存储 在 innodb data home dir 和 innodb data_ file path 表 空间 中 。 


InnoDB 存储 引擎 的 优点 在 于 提供 了 良好 的 事务 管理 、 骨 省 修复 能 力 和 并 发 控制 。 其 缺点 


| 是 读 写 效率 稍 差 ， 占 用 的 数据 空间 相对 比较 大 。 


InnoDB 表 是 如 下 情况 的 理想 引擎 : 

回 ”更 新 密集 的 表 : InnoDB 存储 引擎 特别 适合 处 理 多 重 并 发 的 更 新 请 求 。 

回 ”事务 : InnoDB 存储 引擎 是 唯一 支持 事务 的 标准 MySQL 存储 引擎 ， 这 是 管理 敏感 数 
据 ( 如 金融 信息 和 用 户 注册 信息 〉 的 必需 软件 。 

回 ”自动 灾难 恢复 : 与 其 他 存储 引擎 不 同 ，InnoDB 表 能 够 自动 从 灾难 中 恢复 。 虽 然 
MyISAM 表 是 能 在 灾难 后 修复 ， 但 其 过 程 较 长 。 

Oracle 的 InnoDB 存储 引擎 广泛 应 用 于 基于 MySQL 的 Web、 电 子 商务 、 金 融 系 统 、 健 康 
护理 以 及 零售 应 用 .因为 mnoDB 可 提供 高 效 的 ACID 独立 性 (Atomicity)、 一 致 性 (Consistency)、 
隔离 性 〈Isolation)、 持 和 久 性 (Durability) 兼容 事务 处 理 能 力 ， 以 及 独特 的 高 性 能 和 具有 可 扩展 
性 的 构架 要 素 。 

另外 ，InnoDB 设计 用 于 事务 处 理应 用 ， 这 些 应 用 需要 处 理 崩 江 恢 复 、 参 照 完整 性 、 高 级 
别 的 用 户 并 发 数 ， 以 及 响应 时 间 超 时 服务 水 平 合同 。 在 MySQL 5.5 中 ,最 显著 的 增强 性 能 是 将 
InnoDB 作为 默认 的 存储 引擎 。 在 MyISAM 以 及 其 他 表 类 型 依然 可 用 的 情况 下 ， 用 户 无 须 更 改 
配置 就 可 构建 基于 InnoDB 的 应 用 程序 。 


“126.. 


第 生 章 数据 库 操 作 


3.， MyISAM 和 InnoDB 的 区 别 
(1) InnoDB 不 支持 FULLTEXT 类 型 的 索引 。 
(2) InnoDB 中 不 保存 表 的 具体 行 数 ， 也 就 是 说 ， 当 执行 select count(*) from table 时 ， | 
InnoDB 要 扫描 一 人 遍 整 个 表 来 计算 有 多 少 行 ,但 是 MyISAM 只 要 简单 地 读 出 保存 好 的 行 数 即 可 。 | 天 六 
需要 注意 的 是 ， 当 count(*) 语 句 包 含 where 条 件 时 ， 两 种 表 的 操作 是 相同 的 。 | 
(3) 对 于 AUTO_INCREMENT 类 型 的 字段 ，InnoDB 中 必须 包含 只 有 该 字段 的 索引 ， 但 
是 在 MyISAM 表 中 ， 可 以 和 其 他 字段 一 起 建立 联合 索引 。 | 
(4) DELETE FROM table 时 ，InnoDB 不 会 重新 建立 表 ， 而 是 逐 行 地 进行 删除 。 | 
(5) LOAD TABLE FROM MASTER 操作 对 InnoDB 是 不 起 作用 的 ， 解 决 方法 是 首先 把 | 
InnoDB 表 改 成 MyISAM 表 ， 导 入 数据 后 再 改 成 nnoDB 表 ， 但 是 对 于 使 用 的 额外 的 InnoDB | 
特性 〈 如 外 键 )》 的 表 不 适用 。 | 
另外 ，InnoDB 表 的 行 锁 也 不 是 绝对 的 ， 如 果 在 执行 一 个 SQL 语句 时 MySQL 不 能 确定 要 | 
扫描 的 范围 ，InnoDB 表 同 样 会 锁 全 表 ， 如 update table set num=l where name like “%mr%”。 | 
任何 一 种 表 都 不 是 万 能 的 ， 只 用 恰当 的 针对 业务 类 型 来 选择 合适 的 表 类 型 才能 最 大 地 发 挥 | 
MySQL 的 性 能 优势 。 | 


专家 点 评 


在 MySQL 数据 库 中 ，InnoDB 与 MyISAM 数据 存储 引擎 各 有 特色 。 这 两 个 表 既 存在 一 定 
相似 的 联系 ， 也 存在 着 差异 。 在 性 能 上 ， 可 以 说 是 各 有 千秋 ， 所 以 用 户 必须 根据 自己 的 业务 需 | 
要 来 选择 合适 的 引擎 类 型 ， 以 便 提 高 整个 数据 库 系 统 的 性 能 ， 使 其 达到 最 大 程度 的 优化 。 | 


问题 95 如 何在 MySQL 中 实现 多 关键 字 查 询 ? 


问题 讲述 
在 MySQL 中 ， 查 询 是 最 为 常用 的 操作 ， 那 么 如 何 实现 多 关键 字 的 查询 呢 ? 
专家 解答 | 
作为 刚刚 接触 MySQL 数据 库 的 新 手 ， 您 可 能 往往 会 在 如 查询 语句 的 顺序 、 排 序 、 语 法 结 | 
构 等 环节 上 出 现 问 题 。 其 中 ， 如 何 应 用 多 关键 字 查 询 数 据 同 样 也 是 许多 刚刚 接触 MySQL 的 读 | 
者 们 所 困惑 的 问题 之 一 。 实 现 多 关键 字 查 询 的 方法 很 简单 ， 其 命令 结构 如 下 所 示 : | 
select * from table_name where name like %character1% or ...; | 
其 中 ，table_name 是 需要 查询 的 数据 表 名 称 ,“%” 作 为 通配符 ， 用 来 匹配 与 两 个 “%” 中 | 
间 相 匹配 的 字符 串 内 容 ,or 作 为 其 他 关键 字 查 询 的 连接 关键 字 , 一 个 语句 中 也 可 以 存在 多 个 “or” | 
连接 关键 字 ， 用 于 精确 查询 数据 库 表 中 的 数据 内 容 。 | 
专家 点 评 


实现 多 关键 字 查 询 并 不 难 ， 在 查询 语句 中 加 入 “or” 关 键 字 即 可 ， 另 外 ， 也 可 以 应 用 多 个 
“and” 关 键 字 来 精确 唯一 查询 数据 ， 使 返回 结果 更 加 精确 。 | 
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问题 96 ”如 何在 MySQL 中 修改 用 户 密码 和 


7 
会 内 创建 权限 用 户 ? 


问题 请 述 


| 在 MySQL 数据 库 中 ， 如 何 修改 已 存在 的 用 户 密码 ? 如 何 新 增 数据 库 用 户 ， 并 赋予 该 用 户 
| 特定 权限 ? 
| 专家 解答 
| 很 多 刚刚 接触 MySQL 数据 库 的 用 户 ， 在 默认 情况 下 都 会 使 用 系统 用 户 账户 ， 即 “root” 
| 用 户 。 该 用 户 具 有 最 高 管理 权限 。 但 是 ， 为 了 保证 数据 库 数据 的 安全 性 和 稳定 性 ， 需 要 重 置 账 
| 户 密码 ， 当 其 他 用 户 预 申请 访问 数据 库 时 ， 就 必须 通过 管理 权限 用 户 为 访问 申请 者 新 增加 一 个 
| 具有 特定 操作 权限 的 用 户 账 户 ， 以 限制 用 户 的 操作 权限 。 
| 1. 修改 用 户 密码 
| (1) 用 户 可 以 通过 第 三 方 MySQL 管理 工具 来 修改 用 户 密码 ， 如 phpmyadmin 等 。 用 户 只 
| 需要 登录 图 形 化 管理 界面 ， 并 利用 鼠标 单 击 mysql 数据 库 中 的 user 表 ， 然 后 再 应 用 SQL 语句 
| 来 更 改 特定 用 户 的 账户 密码 即 可 。 在 修改 密码 时 ， 必 须 应 用 PASSWORDO 函 数 ， 以 此 来 保证 
| MySQL 用 户 名 的 安全 。 
| (2) 在 命令 提示 符 中 使 用 mysqladmin， 即 在 命令 提示 符 界面 中 输入 如 下 命令 : 
mysqladmin -u root -p oldpassword newpasswd 
| 执行 上 述 命令 后 , 需要 用 户 输入 root 的 原 密码 , 这 样 root 的 密码 将 改 为 newpasswd。 同样 ， 
| 把 命令 里 的 root 改 为 用 户 的 用 户 名 ， 用 户 就 可 以 更 改 之 前 的 密码 了 。 当 然 ， 如 果 用 户 的 
| mysqladmin 无 法 正常 连接 到 服务 器 上 , 或 因 软件 损坏 而 导致 无 法 运行 ， 则 不 能 使 用 该 方法 ， 并 
| 且 mysqladmin 无 法 将 设置 的 密码 清空 。 
| (3) 使 用 以 下 语句 来 修改 数据 库 用 户 密码 ， 其 命令 如 下 : 
INSERT INTO mysql.user (Host,User,Password) VALUES ('%,', ‘mr', PASSWORD('mrsoft')); 
FLUSH PRIVILEGES 
| 此 方法 更 确切 地 说 是 用 户 向 数据 库 中 增加 一 个 新 用 户 ， 用 户 名 为 mr， 密 码 为 mrsoft。 注 意 
| 要 使 用 PASSWORD 函数 ， 然 后 还 要 使 用 FLUSH PRIVILEGES 来 执行 确认 。 


2. 设置 用 户 权 限 
设置 用 户 权限 应 用 的 是 Grant 命令 。 其 具体 语句 如 下 : 

GRANT ALL PRIVILEGES ON** TO mr@localhost IDENTIFIED BY mrsoft WITH GRANT OPTION; 
对 用 户 的 每 一 项 权限 进行 设置 ， 插 入 : 


mysql>INSERT INTO user VALUES(Clocalhost'， 
systeny', PASSWORD('mrsoft’), 'Y', YY 

Y, YY, "YY, ,YY,Y,' 

OO 


了 。128 。 
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说 明 : 

设置 用 户 权 限 的 用 户 账号 ， 必 须 具备 有 与 “root” 相 同 权 限 ， 即 所 有 执行 权限 。 
专家 点 评 | 二 
“root” 用 户 是 MySQL 安装 时 自 带 的 用 户 ， 也 可 以 说 是 权限 最 大 的 用 户 ， 但 是 ， 出 于 对 | 琶 认 
数据 库 安全 性 的 考虑 ， 在 安装 完 数据 库 后 建议 您 修改 密码 ， 另 外 ， 通 过 创建 其 他 账户 ， 分 配 其 
他 用 户 具 有 查询 、 或 单一 操作 的 权限 〈 只 有 具有 增加 、 删 除 或 更 新 权限 的 用 户 账户 )， 从 而 保证 | 
数据 库 系 统 的 安全 性 和 稳定 性 。 | 


问题 97 在 MySQL 中 两 种 索引 Hash 与 B-Tree 的 


区 别 是 什么 ? 
问题 前 述 
在 MySQL 数据 库 中 ，Hash 型 索引 与 B-Tree 型 索引 有 什么 区 别 呢 ? 
专家 解答 


MySQL 支持 Hash 索引 和 B-Tree 索引 ， 但 两 者 究竟 有 什么 区 别 呢 ? 这 个 问题 可 能 会 困扰 | 
许多 用 户 。 下 面 对 两 者 的 区 别 进行 详细 分 析 。 | 
MySQL Hash 索引 结构 具有 一 定 特殊 性 ， 其 检索 效率 非常 高 ， 索 引 的 检索 可 以 一 次 定位 ， | 
不 像 B-Tree 索引 需要 从 根 节点 到 枝 节点 ， 通 过 多 次 IO 访问 ， 最 后 才能 访问 到 页 节点 ， 所 以 | 
Hash 索引 的 查询 效率 要 远 高 于 B-Tree 索引 。 | 
但 是 Hash 索引 的 效率 要 比 B-Tree 高 很 多 ， 为 什么 在 实际 应 用 中 不 能 都 应 用 Hash 索引 而 | 
还 要 使 用 B-Tree 索引 呢 ? 任何 事物 都 是 有 两 面 性 的 , Hash 索引 也 一 样 , 虽然 Hash 索引 效率 高 ，| 
但 是 Hash 索引 由 于 本 身 的 特殊 性 也 带 来 了 很 多 限制 和 次 端 ， 主 要 有 以 下 几 个 方面 : | 
(1) MySQL Hash 索引 仅仅 能 满足 "=","IN" 和 "<=>" 查 询 ， 不 能 使 用 范围 查询 。 | 
由 于 MySQL Hash 索引 比较 的 是 进行 Hash 运算 之 后 的 Hash 值 , 所 以 它 只 能 用 于 等 值 的 过 | 
滤 ， 不 能 用 于 基于 范围 的 过 滤 ， 因 为 经 过 相应 的 Hash 算法 处 理 之 后 的 Hash 值 的 大 小 关系 ， 并 | 
不 能 保证 和 Hash 运算 前 完全 一 样 。 | 
(2) MySQL Hash 索引 无 法 被 用 来 避免 数据 的 排序 操作 。 | 
由 于 MySQL Hash 索引 中 存放 的 是 经 过 Hash 计算 之 后 的 Hash 值 ， 而 且 Hash 值 的 大 小 关系 | 
并 不 一 定 和 Hash 运算 前 的 键 值 完全 一 样 , 所 以 数据 库 无 法 利用 索引 的 数据 来 避免 任何 排序 运算 。 | 
(3) MySQL Hash 索引 不 能 利用 部 分 索引 键 查询 。 | 
对 于 组 合 索引 ，Hash 索引 在 计算 Hash 值 时 是 组 合 索 引 键 合并 后 再 一 起 计算 Hash 值 ， 而 | 
不 是 单独 计算 Hash 值 ， 所 以 通过 组 合 索引 的 前 面 一 个 或 几 个 索引 键 进行 查询 时 ，Hash 索引 也 | 
无 法 被 利用 。 | 
(4) MySQL Hash 索引 在 任何 时 候 都 不 能 避免 表 扫描 。 | 
上 面 已 经 提 到 ，Hash 索引 是 将 索引 键 通过 Hash 运算 之 后 ,将 Hash 运算 结果 的 Hash 值 和 | 
所 对 应 的 行 指针 信息 存放 于 一 个 Hash 表 中 , 由 于 不 同 索 引 键 存在 相同 的 Hash 值 ， 所 以 即使 取 | 
满足 某 个 Hash 键 值 的 数据 的 记录 条 数 ， 也 无 法 从 Hash 索引 中 直接 完成 查询 ， 仍 要 通过 访问 表 | 
中 的 实际 数据 进行 相应 的 比较 ， 并 得 到 相应 的 结果 。 | 

.129 . 人 


eR pg 需 知 道 的 300 个 问题 
(5) MySQL Hash 索引 遇 到 大 量 Hash 值 相 等 的 情况 后 性 能 并 不 一 定 就 会 比 B-Tree 索引 高 。 
对 于 选择 性 比较 低 的 索引 键 ， 如 果 创 建 Hash 索引 ， 那 么 将 会 存在 大 量 记录 指针 信息 存 于 


nn 


| 同一 个 Hash 值 相关 联 。 这 样 要 定位 某 一 条 记录 时 就 会 非常 麻烦 ， 会 浪费 多 次 表 数 据 的 访问 ， 


从 而 造成 整体 性 能 低下 。 


问题 98 应 用 PHP 操作 MySQL 数据 库 时 应 该 注意 
哪些 问题 ? 


问题 讲述 


专家 解答 


在 使 用 PHP 操作 MySQL 数据 库 时 ， 需 要 注意 的 问题 都 有 哪些 ? 


众所周知 ，PHP 和 MySQL 是 一 对 “黄金 搭档 ”， 二 者 具有 良好 的 兼容 性 和 开源 性 。 但 是 ， 


在 使 用 PHP 操作 MySQL 数据 库 时 ， 仍 需要 注意 一 些 问题 。 


1. 命令 结束 使 用 分 号 (;) 
在 使 用 MySQL 时 ， 作 为 命令 结束 的 标志 ， 每 一 行 命令 都 需要 用 分 号 (;) 作 为 结束 符 。 但 是 


| 应 用 PHP 操作 MySQL 数据 库 时 ， 建 议 用 户 把 操作 结束 符 分 号 省 略 掉 ， 例 如 : 


Imysql query ("INSERT INTO tablename (name, sex) VALUES ('$ name', $sex'")"); 
这 是 因为 PHP 也 是 以 分 号 作为 一 行 的 结束 标识 ， 额 外 的 分 号 有 时 会 让 PHP 的 语法 分 析 器 


无 法 判断 哪里 是 PHP 语句 的 结束 符 ， 哪 里 是 MySQL 语句 的 结束 符 ， 所 以 这 里 建议 用 户 省 略 
| MySQL 结束 符 。 在 这 种 情况 下 , 虽然 省 略 了 分 号 , 但 是 在 应 用 PHP 执行 MySQL 命令 时 , PHP 
| 解释 器 会 自动 向 语句 结尾 添加 一 个 结束 符 标记 。 


2. TEXT、DATE、 和 SET 数据 类 型 

MySQL 数据 表 的 字段 必 是 MySQL 支持 的 数据 类 型 .其 中 以 下 几 个 数据 类 型 需要 读者 特别 
TEXT 不 是 一 种 数据 类 型 ， 虽 然 可 能 有 些 书 上 是 这 么 说 的 ， 它 实际 上 应 该 是 “LONG 
VARCHAR ”或 “MEDIUMTEXT” 类 型 。 

DATE 数据 类 型 的 格式 是 YYYY-MM-DD， 例 如 ，1999-12-08。 读 者 可 以 很 容易 地 用 date 
函数 来 得 到 这 种 格式 的 当前 系统 时 间 : date("Y-m-d")， 并 且 ， 在 DATA 数据 类 型 之 间 可 以 作 减 
法 ， 得 到 相差 的 时 间 天 数 ， 其 代码 如 下 所 示 : 

$age = ($current_date - $birthdate); 

集合 SET 是 一 个 有 用 的 数据 类 型 ， 它 和 枚 举 ENUM 有 些 相似 ， 只 不 过 SET 能 够 保存 多 个 
值 ， 而 ENUM 只 能 保存 一 个 值 而 已 。 另 外 ，SET 类 型 最 多 只 能 够 有 64 个 预定 的 值 ， 而 ENUM 
类 型 却 能 够 处 理 最 多 65535 个 预定 义 的 值 。 如 果 需 要 有 大 于 64 个 值 的 集合 ， 该 怎么 办 呢 ? 这 
时 就 需要 定义 多 个 集合 来 一 起 解决 这 个 问题 了 。 

3. 通配符 

SQL 的 通配符 有 两 种 ， 分 别 是 “*” 和 “%”， 分 别 用 于 不 同 的 情况 。 例 如 ， 如 果 用 户 想 看 
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到 数据 库 的 所 有 内 容 ， 可 以 按 如 下 操作 进行 查询 : 
SELECT* FROM dbname WHERE USER ID LIKE '%'; 


从 上 述 语句 中 可 以 看 出 ， 两 个 通配符 都 被 用 到 。 它 们 表示 相同 的 意思 ,“*” 与 “%” 都 是 | 
用 来 匹配 任何 的 字符 串 ， 但 是 它们 用 在 不 同 的 上 下 文中 。“*” 用 来 匹配 字段 名 ， 而 “%” 用 来 匹 | 
配 字段 值 。 此 时 还 有 一 个 不 容易 引起 注意 的 地 方 是 ,“%” 通 配 符 需要 和 LIKE 关键 字 一 起 使 用 。 | 
另外 ， 还 有 一 个 通配符 ， 就 是 下 划 线 “”， 它 代表 的 意思 和 上 面 不 同 ， 是 用 来 匹配 任何 单 | 
个 字符 的 。 | 
4. NOT NULL 和 空 记录 | 
在 应 用 PHP 设计 带 表 单 和 提交 功能 的 页 面 时 ， 如 果 用 户 在 没有 填 任 何 东 西 的 情况 下 提交 | 
表单 内 容 会 如 何 呢 ? 如 果 用 户 确实 需要 一 个 值 ， 那么 可 以 用 客户 端 脚本 或 服务 器 端 脚本 来 进行 | 
数据 验证 ， 这 一 点 在 前 面 已 经 说 过 了 。 但 是 ， 在 数据 库 中 却 是 允许 一 些 字段 被 空 出 来 什么 也 不 | 
填 。 对 此 类 记录 ，MySQL 将 要 为 之 执行 一 些 事情 : 插入 值 NULL， 这 是 默认 的 操作 。 | 
如 果 用 户 在 字段 定义 中 为 之 声明 NOT NULL (在 建立 或 修改 这 个 字段 时 ), MySQL 将 把 这 | 
个 字段 空 出 来 什么 东西 也 不 填 。 对 于 一 个 ENUM 枚 举 类 型 的 字段 ,如果 为 之 声明 NOT NULL，| 
则 MySQL 将 把 枚 举 集 的 第 一 个 值 插入 到 字段 中 。 也 就 是 说 , MySQL 把 枚 举 集 的 第 一 个 值 作为 | 
一 个 值 为 NULL 的 记录 和 一 个 空 记录 是 有 一 些 区 别 的 。% 通 配 符 可 以 匹配 空 记录 ,但 是 却 | 
不 能 匹配 NULL 记录 。 在 某 些 时 候 ， 这 种 区 别 会 造成 一 些 意 想不到 的 后 果 。 建 议 读者 在 定义 字 | 
段 时 ， 任 何 字段 都 应 该 声明 为 NOTNULL。 | 
如 果 用 户 在 加 入 或 修改 一 个 新 的 字段 之 前 ， 数 据 库 中 已 经 有 了 一 些 记录 ， 这 时 新 加 入 的 字 | 
段 在 原来 的 记录 中 的 值 可 能 是 NULL， 也 可 能 为 空 。 这 也 算是 MySQL 的 一 个 Bug， 所 以 在 这 | 
种 情况 下 ， 使 用 SELECT 查询 要 特别 小 心 。 | 


注意 : | 
在 搜索 NULL 时 必须 用 IS 关键 字 ， 而 使 用 LIKE 关键 字 时 是 不 会 正常 发 挥 其 功效 的 。 | 
专家 点 评 


应 用 PHP 操作 MySQL 数据 库 需要 注意 的 问题 不 仅仅 局 限于 上 述 4 点 ， 在 常用 的 操作 中 ， | 

还 存在 诸多 需要 注意 的 问题 ， 如 向 数据 库 中 插入 中 文 产生 乱码 问题 、 循 环 返 回 数据 内 容 等 , 这 | 

些 都 有 可 能 成 为 用 户 开发 过 程 中 的 绊脚石 。 面 对 一 些 经 常会 遇 到 的 问题 ， 用 户 应 该 学 会 在 实际 | 

应 用 中 积累 必要 经 验 ， 以 免 在 日 后 的 开发 工作 中 因 遇 到 繁琐 的 数据 库 操作 问题 而 无 从 下 手 。 | 
问题 99 应 用 PHP 向 MySQL 插入 数据 时 ， 


如 何 避 免 产 生 乱 码 ? 
问题 阐述 


在 应 用 PHP 函数 mysql_query0 向 MySQL 数据 库 的 特定 表 中 插入 数据 库 时 ,如何 避 免 在 插 | 
入 数据 时 产生 乱码 ? | 


ss 


Pip 六 预 知道 的 300 个 问题 
专家 解答 


这 类 问题 对 于 开发 经 验 不 足 的 PHP 程序 员 来 说 是 比较 常见 的 问题 ， 这 是 由 于 编码 格式 不 
统一 造成 的 。 一 般 情 况 下 ， 数 据 库 默 认 的 编码 类 型 为 “UTF-8”， 如 果 用 户 编写 的 PHP 文件 编 
人 码 格式 为 “GBK” 或 “GB2312” 类 型 ， 可 能 造成 数据 的 编码 格式 不 统一 ， 当 用 户 使 用 其 他 编 
码 格式 的 PHP 文件 ， 向 数据 库 插入 文字 内 容 (一般 为 汉字 )， 应 用 如 phpMyAdmin 等 第 三 方 软 
件 查看 数据 时 ， 会 在 该 数据 库 中 显示 乱码 。 如 果 更 改 数据 库 的 编码 格式 ， 则 可 能 会 造成 前 台 页 


| 面 显 示 不 正常 。 为 了 解决 这 一 问题 ， 用 户 可 以 统一 数据 库 与 前 台 页 面 的 编码 格式 。 


(1) 当 用 户 按照 原来 的 方式 通过 PHP 存 取 MySQL 数据 库 时 ， 即 使 设置 表 的 默认 字符 集 


为 UTF-8 并 且 通 过 UTF-8 编码 发 送 查询 ， 此 时 会 发 现存 入 数据 库 的 仍然 是 乱码 。 问 题 就 出 在 
| connection 连接 层 上 。 解 决 方法 是 在 发 送 查 询 指令 前 执行 一 下 如 下 命令 : 


SET NAMES "utf8'; 
这 句 命 令 语句 相当 于 下 面 的 3 句 指令 : 


SET character_set_client = utf8; 

SET character_set_results = utf8; 

SET character_set_connection = utf8; 

We ” » 和 库 : db_kn， e 
(2) 应 用 第 三 方 软件 设置 MySQL 编码 格 i 


| 式 ， 这 里 以 phpMyAdmin 为 例 。 首 先 ， 打 开 FS 一 一 


| phpMyAdmin 软件 。 选 择 指定 数据 库 中 的 数据 ee 
| 表 ， 单 击 表 “结构 ” 选择 修改 。 在 “整理 ”一 Po 
| 栏 的 下 拉 菜 单 中 选择 匹配 的 编码 格式 ， 其 结果 Lo 
| 如 图 4.1 所 示 。 We 


| 加 入 如 下 语句: 


然后 在 PHP 处 理 页 的 操作 数据 库 语 名 后 面 图 41 应 用 phpMyAdmin 修改 数据 表 编码 格式 


mysql_query("set names utf-8"); 
这 样 ， 在 应 用 PHP 向 MySQL 数据 库 添 加 内 容 ， 就 不 会 因 编码 不 统一 而 导致 出 现 乱码 。 


| 专家 点 评 


虽然 MySQL 与 PHP 被 称 作 “黄金 组 合 ”， 但是， 对 于 编码 格式 的 统一 问题 ， 读 者 还 是 需 
要 多 加 注意 ， 因 为 一 些 不 能 预期 的 错误 往往 都 发 生 在 编码 格式 的 交互 上 。 所 以 ， 即 便 是 经 验 丰 
富 的 程序 员 ， 在 开发 过 程 中 也 应 该 提高 警惕 ， 和 避免 因 数据 库 编码 格式 和 处 理 页 编码 格式 不 统一 
而 造成 程序 错误 。 


问题 100 MySQL 常见 错误 代码 有 哪些 ? 
问题 阐述 


在 操作 MySQL 过 程 中 ， 由 于 操作 不 当 或 输入 命令 的 格式 不 当 等 原因 ， 经 常会 出 现 一 些 错 
误 或 敬告， 常见 的 错误 代码 有 哪些 ? 分 别 代 表 的 错误 信息 是 什么 ? 
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MySQL 返回 的 错误 代码 可 以 提示 用 户 错误 类 型 ， 并 帮助 用 户 改正 错误 ， 用 户 可 以 通过 常 | 
用 的 错误 代码 判断 错误 原因 。 其 中 常用 的 错误 代码 和 相关 解释 如 下 所 示 : | 

130: 文件 格式 不 正确 ， 可 以 尝试 使 用 repairphp 来 修复 ， 如 果 不 行 就 只 能 恢复 相关 的 数据 | 
了 “此 时 不 是 很 清楚 错误 的 状况 )。 | 

145: 文件 无 法 打开 ， 使 用 repairphp 修复 。 

1005: 创建 表 失 败 。 

1006: 创建 数据 库 失 败 。 

1007: 数据 库 已 存在 ， 创 建 数据 库 失 败 。 

1008: 数据 库 不 存在 ， 删 除数 据 库 失 败 。 

1009: 不 能 删除 数据 库 文 件 而 导致 删除 数据 库 失 败 。 

1010: 不 能 删除 数据 目录 而 导致 删除 数据 库 失 败 。 

1011: 删除 数据 库 文 件 失败 。 

1012: 不 能 读 取 系 统 表 中 的 记录 。 

1016: 文件 无 法 打开 ， 使 用 后 台 修复 或 者 使 用 phpMyAdmin 进行 修复 。 

1017: 服务 器 非法 关机 ， 导 致 该 文件 损坏 。 

1020: 记录 已 被 其 他 用 户 修改 。 

1021: 硬盘 剩余 空间 不 足 ， 需 加 大 硬盘 可 用 空间 。 

1022: 关键 字 重复 ， 更 改 记录 失败 。 

1023: 关闭 时 发 生 错误 。 

1024: 读 文 件 错误 。 

1025: 更 改名 字 时 发 生 错误 。 

1026: 写 文件 错误 。 

1030: 可 能 是 服务 器 不 稳定 〈 具 体 原因 不 是 很 清楚 )。 

1032: 记录 不 存在 。 

1036: 数据 表 是 只 读 的 ， 不 能 对 它 进行 修改 。 

1037: 系统 内 存 不 足 ， 需 重启 数据 库 或 重启 服务 器 。 

1038: 用 于 排序 的 内 存 不 足 ， 需 增 大 排序 缓冲 区 。 

1040: 已 到 达 数 据 库 的 最 大 连接 数 ， 需 加 大 数据 库 可 用 连接 数 。 

1041: 系统 内 存 不 足 。 

1042: 无 效 的 主机 名 。 

1043: 无 效 连接 。 

1044: 数据 库 用 户 权限 不 足 ， 需 联系 空间 商 解 决 。 

1045: 数据 库 服务 器 /数据 库 用 户 名 /数据 库 名 /数据 库 密码 错误 ， 需 联系 空间 商检 查账 户 。 

1046: 没有 选择 数据 库 。 

1048: 字段 不 能 为 空 。 

1049: 数据 库 不 存在 。 

1050: 数据 表 已 存在 。 

1051: 数据 表 不 存在 。 

1054: 字段 不 存在 或 程序 文件 与 数据 库 有 冲突 ， 需 使 用 正确 的 程序 文件 上 传 并 覆盖 。 

1060: 字段 重复 ， 导 致 无 法 插入 这 个 字段 。 

1062: 字段 值 重复 ， 入 库 失 败 。 


. 133 L_ 


1064: 
1065: 
1081: 
1114: 
1115:; 
1116: 
1129: 
1130: 
1133: 
1135: 
1141: 
1142: 
1143: 
1146: 
1147: 
1149: 
1169: 
1177: 
1180: 
1181: 
1193: 


人 Pp 六 预 知道 的 300 个 问题 


MySQL 不 支持 错误 提示 中 的 编码 。 
无 效 的 SQL 语句 ，SQL 语句 为 空 。 
不 能 建立 Socket 连接 。 

数据 表 已 满 ， 不 能 容纳 任何 记录 。 
设置 的 字符 集 在 MySQL 并 没有 支持 。 
打开 的 数据 表 太 多 。 

数据 库 出 现 异 常 ， 需 重启 数据 库 。 
连接 数据 库 失败 ， 没 有 连接 数据 库 的 权限 。 
数据 库 用 户 不 存在 。 

可 能 是 内 存 不 足够 ， 需 联系 空间 商 解决 。 
当前 用 户 无 权 访 问 数据 库 。 

当前 用 户 无 权 访问 数据 表 。 

当前 用 户 无 权 访问 数据 表 中 的 字段 。 
数据 表 缺 失 ， 需 恢复 备份 数据 。 

未 定义 用 户 对 数据 表 的 访问 权限 。 
SQL 语句 语法 错误 。 

字段 值 重复 ， 更 新 记录 失败 。 

打开 数据 表 失 败 。 

提交 事务 失败 。 

回 滚 事务 失败 。 

不 支持 字符 集 限 定 (SETNAMES)。 


1203: 当前 用 户 和 数据 库 建 立 的 连接 已 到 达 数 据 库 的 最 大 连接 数 ， 需 增 大 可 用 的 数据 库 连 
接 数 或 重启 数据 库 。 

1205: 加 锁 超时 。 

1211: 当前 用 户 没有 创建 用 户 的 权限 。 


1267: 
2002: 
2003: 


不 合法 的 混合 字符 集 。 
服务 器 端口 不 正确 ， 需 咨询 空间 商 以 获取 正确 的 端口 。 
MySQL 服务 未 启动 ， 需 启动 该 服务 。 


2013: 远程 连接 数据 库 有 时 会 出 现 这 个 问题 ， 是 MySQL 服务 器 在 执行 一 条 SQL 语句 时 
失去 了 连接 造成 的 。 


专家 点 评 


常用 的 错误 代码 信息 可 以 返回 系统 提示 的 错误 类 型 ， 同 时 ， 该 错误 代码 也 可 以 帮助 用 户 对 
| 不 同 的 错误 或 异常 及 时 做 出 相应 处 理 。 了 解数 据 库 的 常用 错误 代码 ， 对 于 程序 开发 者 或 数据 库 
| 管理 员 来 说 是 非常 重要 的 ， 这 样 可 以 使 开发 和 维护 人 员 更 快 、 更 准确 地 找 出 错误 原因 。 


”问题 阐述 


问题 101 在 MySQL 中 ， 如 何 实现 多 表 查 询 ? 


为 了 方便 查看 多 个 数据 表 内 容 ， 如 何 通 过 SQL 语句 实现 多 表 查 询 ? 
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专家 解答 


在 实际 项 目 开 发 过 程 中 ， 经 常 需要 将 不 同 的 信息 存储 在 不 同 的 表 中 ,， 表 与 表 之 间 通过 某 字 | 
段 互相 联系 ， 从 而 使 表 的 指针 形成 一 种 联动 关系 ， 从 而 可 以 通过 SQL 语言 的 SELECT 语句 实 | 
现 多 表 查 询 。 在 SQL 语言 中 ， 通 过 where 子 句 实现 多 表 查 询 ， 所 要 查找 的 字段 名 最 好 以 “ 表 | 
名 .字段 名 ”格式 表示 ， 这 样 可 以 防止 因 表 之 间 字 自重 名 而 造成 无 法 获知 字段 属于 哪个 表 。 在 ， 
where 子 句 中 多 个 表 之 间 所 形成 的 联动 关系 应 按 “ 表 1 字段 = 表 2. 字 段 and 其 他 查询 条 件 ” 形 式 
书写 。 综 上 所 述 ， 实 现 多 表 查 询 应 按 如 下 形式 书写 : | 

SELECT 字段 名 from 表 1, 表 2...where 表 1. 字段 , 表 2. 字 段 and 其 他 查询 条 件 ; 


另外 ， 为 了 提高 SQL 语句 的 可 读 性 ， 经 常 采 取 为 表 取 别 名 的 方式 对 表 进行 查询 。 使 用 表 | 
别名 可 以 通过 两 种 方式 为 表 指定 别名 。 | 
在 MySQL 中 ， 可 以 通过 关键 字 as 指定 返回 查询 结果 的 表 别 名 ， 其 语法 结构 如 下 : 
SELECT 字段 名 from 表 名 as 表 别 名 where 条 件 …: | 
应 用 as 关键 字 为 一 般 形式 ， 其 实现 过 程 就 像 将 两 张 表 上 的 数据 表 插入 第 三 张 表 中 ， 并 将 | 
返回 的 数据 组 合成 表 展现 给 用 户 。 
在 表 名 后 直接 加 表 的 别名 从 而 指定 表 的 别名 。 其 语法 结构 如 下 : 
SELECT 字段 名 from 表 名 表 别 名 where 条 件 .…; 
多 表 查 询 的 示例 如 下 所 示 
SELECT 字段 名 from students as stu, studentinfo as stu_info where stu.id=stuinfo.sid; | 
另外 ,在 SQL 语言 中 ,使 用 union 或 all 关键 字 可 以 将 多 张 表 的 信息 合并 输出 。 其 中 union 
关键 字 可 以 将 多 个 select 语句 的 查询 结果 合并 输出 , 并 且 可 以 删除 重复 行 , 而 使 用 all 关键 字 可 | 
以 将 多 个 select 语句 的 查询 结果 合并 输出 ， 但 不 删除 重复 行 。 | 


加 


注意 : | 

别名 通常 是 一 个 缩短 的 表 名 ， 用 于 在 连接 中 引用 表 中 的 列 名 ， 如 果 连 接 中 的 多 个 表 中 有 相 | 
同 的 名 称 列 存在 ， 则 要 求 必 须 使 用 表 名 或 表 的 别名 限定 列 名 。 另 外， 如 果 定 义 了 表 的 别名 就 不 | 
能 使 用 表 名 。 | 


专家 点 评 


在 实际 数据 库 查询 过 程 中 ,往往 会 查询 多 个 数据 表 中 的 多 个 字段 ,或 者 在 查询 过 程 中 形成 
内 联 或 相互 制约 条 件 ， 逐 条 地 查询 不 仅 费 时 费力 ， 在 查询 过 程 中 也 不 直观 。 通 过 多 表 查 询 ， 可 | 
以 向 用 户 展示 用 户 所 需要 查看 的 列表 及 数据 ， 从 而 避免 不 必要 的 麻烦 。 | 


问题 102 ”在 MySQL 中， 如 何 实现 内 连接 查询 ? 
问题 阐述 | 
在 实际 开发 过 程 中 ， 经 常 需要 从 多 个 表 中 查询 信息 ， 在 MySQL 数据 库 中 ， 如 何 通过 内 连 | 

接 实现 多 表 查 询 呢 ? 


"135。 


司 Pp 六 须 知道 的 300 个 辣 是 


内 连接 查询 大 多 数 都 比较 复杂 ， 如 对 多 张 表 通过 内 连接 方式 建立 关联 后 ， 还 需要 在 SQL 

语句 中 再 指定 一 些 查询 条 件 ， 从 而 进一步 实现 查询 的 精确 定位 。 内 连接 查询 的 基本 语法 如 下 : 
selectfieldlist from tablel[inner] join table2 on table.1column1=table2.column?; 

其 中 fieldlist 为 要 查询 的 字段 列表 ，tablel 和 table2 为 要 连接 的 表 名 ，inner 表示 表 之 间 的 


| 连接 方式 为 内 连接 ,tablel.column1=table2.column2 用 于 指明 表 tablel 和 table2 之 间 的 连接 条 件 。 


一 般 情况 下 , 复杂 的 内 连接 查询 是 在 基本 内 连接 查询 的 基础 上 再 附加 一 些 查询 条 件 , 例如 ， 


| 下 面 语句 就 是 一 个 比较 复杂 的 内 连接 语句 。 其 代码 如 下 ， 


select tb yg,.userid,tb yg.name,tb yg.sex,tb yg.bm from tb yg inner join tb yg info on tb yg.userid= 


| tb yg_info.ygid where tb_yg.userid="yg001’; 


上 述 语句 中 通过 where 子 句 指定 要 查询 的 员工 编号 ， 最 终 实现 内 连接 表 的 查询 。 


| 专家 点 评 


在 实际 应 用 中 ， 通 过 内 连接 实现 多 表 查 询 应 用 非常 广泛 。 为 了 简化 查询 步 又 ， 用 户 在 实际 


| 应 用 中 可 以 使 用 内 连接 查询 数据 .一般 情 况 下 , 不 但 可 以 应 用 where 子 句 , 也 可 以 应 用 如 having 
| 子 句 、order by 子 句 实现 内 连接 查询 。 


问题 103 在 MySQL 中 ， 如 何 实现 外 连接 查询 ? 


问题 阐述 


| 专家 解答 


要 取决 于 所 建立 的 外 连接 的 类 型 。 在 实际 应 用 中 常 采用 “outer join” 关 键 字 实 现 外 连接 ， 其 中 


为 了 从 多 个 表 中 查询 符合 条 件 信息 ， 在 MySQL 数据 库 中 ， 如 何 通 过 外 连接 实现 部 分 或 全 
部 匹配 结果 集 。 


内 连接 返回 的 是 两 个 或 多 个 表 中 符合 条 件 的 数据 ， 而 外 连接 返回 部 分 或 全 部 匹配 行 ， 这 3 


出 


要 包含 连接 的 两 个 表 中 的 不 匹配 行 ， 可 以 通过 左 连接 和 右 连接 方式 实现 。 
应 用 外 连接 的 一 般 查 询 语法 结构 如 下 : 
select * from tablenamel left/right outer join tablename2 on tablenamel. column1=tablename2. column2 


tablenamel 、tablename2 是 需要 连接 的 数据 表 , lefvright 用 于 指定 该 外 连接 是 应 用 左 连 接 或 
右 连 接 方式 将 数据 表 tablenamel 和 数据 表 tablename2 进行 连接 。 

上 述 结构 中 , 提 到 应 用 左 连接 或 有 连接 方式 实现 两 个 表 中 的 不 匹配 行 。 下 面 以 左 连接 为 例 ， 
向 读者 展示 如 何 应 用 左 连接 返回 查询 结果 。 

左 连接 返回 的 查询 结果 包含 坐标 中 的 所 有 符合 查询 条 件 及 右 表 中 所 有 满足 连接 条 件 的 行 ， 
MySQL 数据 库 中 使 用 左 连接 的 语法 格式 如 下 : 

select fieldl[field2...] from tablel left [outer] join table on join_condition[where search_condition]; 
其 中 lefit outer join 表示 表 之 间 通 过 做 连接 方式 相互 连接 ， 也 可 以 简写 成 left join; on 
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join_condition 值 多 表 建 立 连 接 所 使 用 的 连接 条 件 ; where search_condition 为 可 选项 ， 用 于 设置 | 


查询 条 件 。 
专家 点 评 


与 左 连接 (left outer join) 类 似 , 右 连接 可 以 将 右 表 中 满足 查询 条 件 的 记录 和 坐标 中 满足 连接 
条 件 的 记录 全 部 显示 出 来 。 对 于 刚刚 接触 MySQL 数据 库 的 用 户 来 说 ， 这 可 能 会 稍微 花费 他 们 | 
的 一 点 时 间 来 了 解 这 些 内 容 。 应 用 外 连接 同样 可 以 实现 连接 匹配 并 拼接 两 个 数据 表 中 的 相关 字 | 


段 ， 这 样 可 以 使 用 户 在 查询 过 程 中 ， 更 清晰 地 查看 所 需要 访问 的 数据 内 容 。 


问题 104 在 MySQL 中 ， 如 何 实现 党 套 查 询 ? 


问题 阐述 

嵌 套 查询 在 数据 库 系统 开发 过 程 中 应 用 广泛 ， 如 何在 MySQL 中 利用 嵌 套 查询 ， 来 提高 复 
杂 的 查询 语句 可 读 性 呢 ? | 
专家 解答 


嵌 套 查询 在 数据 库 系统 开发 中 可 以 对 复杂 查询 进行 分 解 ， 从 而 提高 查询 语句 的 可 读 性 。 其 


中 复杂 查询 可 以 分 为 简单 的 嵌 套 查询 和 复杂 的 嵌 套 查询 。 


下 面 通过 一 个 简单 的 示例 向 读者 介绍 如 何 应 用 SQL 语句 的 嵌 套 ， 来 实现 翌 套 查询 。 子 查 | 
询 是 一 个 select 语句 ， 返 回 单个 值 且 嵌 套 在 select、insert、update 和 delete 语句 中 。 任 何 可 以 | 
使 用 表达 式 的 地 方 都 可 以 应 用 子 查询 作为 嵌 套 查询 条 件 。 其 中 子 查询 也 称 作 内 部 查询 或 内 部 连 | 
接 ， 包 含 子 查询 的 语句 成 为 外 部 查询 或 外 连接 ， 这 些 都 可 以 应 用 到 嵌 套 查询 中 。 简 单 的 霸 套 查 | 


询 语句 如 下 : 


select * from tablenamel where sid in(select sid from tablename2 where tid=num); 


其 中 tablenamel 和 tablename2 分 别 是 被 查询 的 数据 表 ;“sid” 与 “tid” 作 为 子 查 询 限定 条 件 。 
另外 ， 在 实际 的 数据 库 系统 开发 中 ， 对 两 张 表 实现 嵌 套 相对 容易 ， 对 多 表 实 现 嵌 套 查询 相 | 
对 麻烦 一 些 。 一 般 情况 下 ， 可 以 连续 在 查询 语句 的 子 句 中 幅 套 select 语句 。 其 基本 的 形式 如 下 | 


所 示 : 


Select * from tablenamel where sid in (select sid from tablename2 where name in (select...)); 


另外 ， 实 现 多 表 的 岩 套 时 ， 可 以 同时 使 用 谓词 ， 如 any、some、all， 这 些 谓词 可 以 和 比较 | 
运算 符 一 起 使 用 ， 判 断 是 否 任何 或 全 部 返回 值 都 满足 搜索 条 件 。 其 中 some 和 any 是 存在 量 的 ，| 
判断 返回 值 是 否 满足 搜索 条 件 ， 这 两 个 词 含义 相同 ， 可 以 替换 使 用 ，all 谓词 称 为 通用 谓词 , 它 | 


只 关心 是 否 有 谓词 都 满足 搜索 条 件 。 其 中 谓词 的 语法 形式 如 下 : 
scalar_ expression{ 查 询 条 件 运 算 符 } {somelany|}(subquery) 
其 中 scalar_expression 指 任何 有 效 的 SQL 表达 式 ; subquery 指 包含 某 列 结果 集 的 子 查询 。 
专家 点 评 


嵌 套 查询 在 实际 项 目 开发 过 程 中 应 用 比较 繁琐 ， 它 可 以 通过 一 条 SQL 语句 实现 多 表 之 间 
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i 
的 复杂 查询 , 与 子 查 询 不 同 的 是 , 子 查 询 中 只 是 通过 本 数据 表 的 相关 字段 内 容 来 精确 查询 条 件 ， 
而 嵌 套 查询 可 能 会 从 其 他 数据 表 中 获取 其 他 字段 名 称 进行 查询 。 男 外 ， 嵌 套 查 询 中 可 能 也 包含 
子 查询 条 件 。 与 连接 查询 不 同 的 是 ， 用 户 需 要 查询 的 是 某 个 表 的 数据 ， 而 不 是 将 几 个 数据 表 连 
接 后 再 显示 的 数据 ， 所 以 在 实际 应 用 过 程 中 一 定 要 灵活 运用 其 套 查询 。 


问题 105 如何 连接 远程 MySQL 服务 器 ? 


| 问题 阐述 


在 实际 的 生产 环境 中 ， 我 们 所 架设 的 PHP 程序 运行 环境 ， 多 数 情况 下 Apache 服务 器 与 


MySQL 服务 器 是 不 放置 在 同一 台 服 务 器 中 的 ， 这 样 就 会 涉及 一 个 问题 ， 即 如 何 实现 程序 与 
| MySQL 服务 器 的 连接 ? 如 果 不 能 够 很 好 地 实现 与 MySQL 服务 器 的 连接 , 其 他 任何 操作 都 无 
| 从 谈 起 。 


| 专家 解答 


在 连接 远程 MySQL 服务 器 之 前 ， 首 先 ， 应 该 为 远程 服务 器 创建 一 个 远程 登录 用 户 , 然后 ， 


| 以 远程 登录 用 户 的 身份 登录 服务 器 , 为 将 要 访问 这 个 远程 MySQL 服务 器 的 卫 地 址 开通 一 个 用 


户 ， 最 后 就 可 以 通过 这 个 用 户 来 访问 远程 的 服务 器 了 。 
例如 ，MySQL 服务 器 安装 于 Tp 为 192.168.1.194 的 机 器 上 ， 项 目 存储 于 192.168.1.59 的 机 
器 上 ， 下 面 就 在 192.168.1.59 的 机 器 上 连接 192.168.1.194 机 器 上 的 MySQL 服务 器 。 其 具体 操 
作 步 又 如 下 : 
(1) 在 192.168.1.194 的 机 器 上 ， 以 localhost 服务 器 、root 用 户 的 身份 登录 MySQL 服务 
器 , 为 MySQL 服务 器 添加 一 个 用 户 , 设置 用 户 名 为 root, 服务 器 为 192.168.1.194, 密码 为 111， 
其 拥有 所 有 的 权限 。 在 phpMyAdmin 图 形 化 管理 工具 中 添加 用 户 的 操作 如 图 4.2 所 示 。 


国 服务 器 ; localhost 
东部 提 库 如 SQL 惠 状 杰 。 问 变星 困 字 符 集团 Engines ”全 权限 惫 进程 下 导出 本 Import 


最 务 器 ，192.168.1.194 
用 户 各，root 
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图 4.2 添加 数据 库 用 户 (一 ) 
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Je 

(2) 在 192.168.1.194 的 机 器 上 ， 以 新 添加 的 用 户 (192.168.1.194 服务 器 、root 用 户 、 密 | 
码 111) 身份 登录 MySQL 服务 器 ， 并 为 该 服务 器 添加 一 个 新 用 户 ， 同 时 设置 用 户 名 为 mr, 服 | 
务 器 为 192.168.1.59， 密 码 为 mrsoft， 其 拥有 数据 的 添加 、 修 改 、 删 除 和 查询 的 权限 。 在 | 
phpMyAdmin 图 形 化 管理 工具 中 添加 用 户 的 操作 如 图 43 所 示 。 | 


本 服务 192168.1194 
人 诈 ” 开 :sok 起 杖 吉 辐 丈 里 开 字 和 时 同 cree 和 本 雹 和 总 3meor 
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| 
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图 43 添加 数据 库 用 户 〈 二 ) | 
(3) 在 192.168.1.59 的 机 器 上 ， 在 PHP 项 目 中 定义 连接 远程 MySQL 服务 器 的 用 户 名 为 | 
mr， 密 码 为 mrsoft， 设 置 服务 器 为 192.168.1.59， 成 功 连 接 远程 MySQL 服务 器 。 其 代码 如 下 : | 


<2php 
$conn=mysql_ connect("192.168.1.194","mr","mrsoft") or die(mysql_errorO); 
?> 


注意 : | 
在 连接 远程 的 MySQL 服务 器 时 必须 关闭 MySQL 服务 器 所 在 机 器 的 防火 墙 , 否则 MySQL | 
服务 器 是 不 允许 远程 连接 的 。 | 


专家 点 评 


在 实际 的 生产 环境 中 ， 项 目 和 数据 库 的 分 隔 存储 是 一 个 非常 好 的 方案 ， 不 但 能 够 提高 其 安 | 
全 性 ， 而 且 还 可 以 降低 服务 器 被 攻击 后 的 损失 程度 。 上 述 介绍 的 是 通过 phpMyAdmin 图 形 化 管 | 
理工 具 来 创建 远程 MySQL 服务 器 用 户 ， 在 通过 phpMyAdmin 图 形 化 管理 工具 管理 MySQL 数 | 
据 库 时 有 一 项 非常 重要 的 设置 , 即 登录 时 所 输入 的 用 户 名 和 密码 , 在 phpMyAdmin 图 形 化 管理 | 
工具 的 配置 文件 《config.inc php) 中 默认 设置 的 是 localhost 服务 器 ， 用 户 名 为 root， 密 码 为 空 。 | 
如 果 用 户 想 要 通过 192.168.1.194 的 服务 器 ， 以 root 用 户 、 密 码 111 的 身份 进行 登录 ,那么 必须 | 
对 配置 文件 (config.inc.php)〉 进 行 修改 ， 修 改 内 容 如 图 4.4 所 示 。 | 

在 完成 上 述 修改 之 后 ， 即 能 够 以 192.168.1.194 服务 器 ， 用 户 名 root，, 密码 111 的 身份 登录 | 
到 MySQL 服务 器 。 需 要 注意 的 是 ， 只 有 以 这 个 身份 登录 ， 为 指定 的 了 P 地 址 以 及 对 应 的 用 户 授 | 
权 后 ， 这 个 对 应 的 用 户 才 具备 远程 访问 的 权限 。 | 


人 9is 


图 4.4 修改 配置 文件 oy 
问题 106 ”如 何 实 现 MySQL 数据 库 的 双 机 热 备份 ? 


| 问题 阐述 
| 在 Web 项 目 中 其 最 薄弱 的 环节 就 是 它 的 服务 器 , 一 旦 服务 器 受到 攻击 ,就 将 面临 内 容 被 删 
， 除 、 数 据 被 算 改 的 危险 ， 更 有 其 者 会 导致 服务 器 瘫痪 。 为 了 保证 数据 的 安全 ， 对 数据 进行 备份 
| 是 必要 的 。 备 份 数 据 最 佳 的 方法 就 是 实现 数据 库 的 双 机 备份 ， 即 存在 两 个 数据 库 服务 器 ， 当 程 
| 序 执行 一 次 操作 时 ， 同 时 将 数据 保存 到 两 个 数据 库 服务 器 中 ， 从 而 确保 数据 的 安全 性 ， 即 使 有 
| 一 个 数据 库 服务 器 瘫痪 ， 那 么 还 有 另外 一 个 数据 库 服务 器 可 以 使 用 。 我们 这 里 所 要 解决 的 问题 
| 就 是 如 何 实 现 MySQL 数据 库 的 双 机 热 备份 。 

| 专家 解答 

| 双 机 热 备份 通过 日 志文 件 来 传输 服务 器 上 数据 的 变化 。 主 服务 器 进程 在 数据 被 更 新 时 触 
| 发 ， 并 将 相应 的 日 志文 件 发 送 到 从 服务 器 ， 从 服务 器 进程 接收 到 主 服 务 器 发 送 的 日 志文 件 后 ， 
| 阅读 日 志文 件 并 对 响应 的 数据 库 进 行 操作 。 具 体 的 工作 流程 如 图 45 所 示 。 


2 让 服务 器 
日 志 发 送 进程 
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@ 
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器 接收 进程 。 人 服务 可 《 日 玉 访 到， 
进程 读 取 日 志 后 更 新 ) 


让 服务 器 
图 4.5 双 机 热 备份 的 工作 原理 


1. 主 服 务 器 的 配置 
主 服 务 器 的 下 设置 为 192.168.1.194。 配 置 主 服务 器 的 步骤 如 下 : 
| (1) 找到 MySQL 数据 库 的 配置 文件 myini。 该 文件 可 能 存储 于 Windows 的 安装 目录 或 
| MySQL 的 安装 目录 之 下 。 
(2) 通过 编辑 器 打开 myini 文件 ， 定 位 到 [mysqld] 一 节 ， 并 在 该 节 中 加 入 如 下 内 容 
server-id=] 
log-bin=c:/log-bin.log 
binlog-do-db=dbname 
. 140. 
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其 中 server-id 指定 服务 器 编号 ; log-bin 为 日 志 的 存储 位 置 及 文件 名 ; binlog-do-db 为 进行 ， 

日 志 记 录 的 数据 库 名 称 ， 如 果 存 在 多 个 数据 库 ， 则 用 逗号 “, ”分 隔 | 
例如 ， 指 定 服务 器 编号 为 1， 日 志 的 存储 位 置 和 名 称 是 C 生根 目录 下 log-binlog， 指 定 对 | 

db backup 数据 库 进 行 日 志 记录 操作 。 | 
(3) 创建 db_backup 数据 库 。 | 

(4) 进入 MySQL 的 命令 操作 界面 ， 通 过 命令 为 从 服务 器 授权 访问 数据 库 的 用 户 名 和 密 | 

人 码 。 命 令 如 下 : 


grant replication slave on *.* to ‘mr'(@'192.168.1.96' identified by ‘mrsoft' ; | 
其 中 指定 用 户 名 是 mr， 从 服务 器 的 卫 是 192.168.1.96， 密 码 是 mrsoft。 命 令 执行 的 操作 界 | 
面 如 图 4.6 所 示 。 | 
(5) 重启 MySQL 服务 器 ， 使 更 改 生效 。 
(6) 重新 进入 MySQL 的 命令 操作 界面 ， 执 行 如 下 命令 检测 配 否 生 效 。 
show master status \G 
若 出 现 如 图 4.7 所 示 的 结果 ， 则 说 明 配 置 成 功 。 


th192. 100.1. 194 “PHL 


图 4.6 执行 为 从 服务 器 授权 的 操作 图 4.7 检测 MySQL 主 服 务 器 配置 
2， 从 服务 器 的 配置 

(1) 在 从 服务 器 中 创建 与 主 服务 器 中 相同 的 数据 讨 

(2) 找到 从 服务 器 中 的 my.ini 文件 ， 定 位 到 [mysqld] 一 节 ， 并 在 该 节 中 加 入 如 下 内 容 : 
server-id=2 
master-host=192.168.1.194 
master-port=3306 
master-user=mr 


master-password=mrsoft 
master-connect-retry=60 


其 中 server-id 指定 从 服务 器 的 编号 ， 该 编号 必须 与 主 服务 器 不 同 ，master-host 指定 主 服 务 | 
器 的 卫 地 址 ;master-port 指定 访问 主 服务 器 所 用 端口 号 ; master-user 和 master-password 指定 | 
访问 主 服务 器 的 用 户 名 和 密码 ，master-connect-retry 表示 如 果 因 为 网 络 问题 与 主 服务 器 断 开 连 | 
接 ， 则 等 待 多少 秒 尝试 重新 连接 。 | 
(3) 重新 启动 MySQL 服务 器 ， 使 配置 生效 。 
(4) 进入 从 服务 器 的 MySQL 命令 操作 界面 ， 执 行 如 下 命令 启动 进程 。 
slave start: 


命令 执行 的 界面 如 图 4.8 所 示 。 
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图 4.8 从 服务 器 启动 进程 
至 此 ，MySQL 数据 库 双 机 热 备份 配置 完成 。 
| 专家 点 评 


| 双 机 热 备份 是 一 种 很 好 的 备份 数据 库 的 方法 ， 在 应 用 该 方法 时 ， 有 一 点 必须 要 注意 ， 就 是 
| 必须 关闭 从 服务 器 的 防火 墙 ， 否 则 主 服务 器 中 的 MySQL 数据 是 不 能 添加 到 从 服务 器 的 。 


问题 107 解决 MySQL 语句 错误 的 最 佳 方案 是 什么 ? 


| 问题 阐述 


| 在 应 用 PHP 内 置 函数 操作 MySQL 数据 库 时 ，MySQL 语句 产生 的 错误 是 很 难 发 现 的 ， 因 
| 为 在 PHP 脚本 中 执行 MySQL 的 添加 、 修 改 、 查 询 和 删除 语句 时 ， 它 不 会 输出 任何 的 信息 ， 除 
| 非 对 该 语句 是 否 执行 成 功 进行 了 判断 。 这 也 就 导致 无 法 判断 出 SQL 语句 是 否 被 正确 执行 ， 无 
| 法 捕捉 到 MySQL 语句 的 错误 信息 。 这 就 是 我 们 在 这 里 将 要 解决 的 问题 。 


| 专家 解答 


| 在 讲解 解决 方案 之 前 ， 我 们 有 必要 先 来 了 解 一 下 错误 的 根源 ， 正 所 谓 “ 对 症 下 药 ， 方 可 
| 药 到 病 除 ”。PHP 操作 MySQL 数据 库 出 现 错误 的 地 方 可 以 归结 为 3 个 环节 : 第 一 是 数据 表单 
| 的 提交 ， 第 二 是 数据 的 处 理 ， 第 三 是 存储 到 数据 库 。 这 3 个 环节 是 MySQL 最 容易 产生 错误 的 
| 地 方 。 
| 第 一 个 环节 表单 提交 页 和 数据 处 理 页 之 间 , 在 表单 提交 页 中 输入 的 文本 域名 称 一 定 要 与 提 
| 交 到 数据 处 理 页 的 变量 相 一 致 , 且 大 小 写 必须 统一 , 因为 获取 表单 提交 数据 的 $_GET 和 $_POST 
| 方法 是 区 分 大 小 写 的 。 如 果 不 一 致 ， 那 么 文本 域 的 值 是 不 会 被 存储 到 数据 库 中 的 。 

| 第 二 个 环节 是 数据 处 理 页 ， 在 该 页 的 SQL 语句 中 字段 名 称 排列 的 顺序 与 提交 的 变量 值 的 
| 顺序 要 一 一 对 应 。 如 果 顺 序 不 对 应 ， 由 于 字段 的 类 型 不 同 ， 将 导致 数据 不 能 正常 添加 。 

| 第 三 个 环节 是 数据 处 理 页 和 数据 表 之 间 ， 在 数据 处 理 页 编写 的 SQL 语句 中 使 用 的 字段 名 
| 称 一 定 要 与 数据 表 中 的 字段 名 称 相同 。 

| 了 解 了 问题 的 根源 后 ， 下 面 看 解决 方案 。 如 何 才能 准确 地 捕捉 到 这 个 错误 信息 ， 最 佳 的 方 
| 案 就 是 应 用 mysql_errorO 函 数 返 回 错误 信息 。 该 函数 在 mysql_query0 函 数 后 应 用 ， 能 够 准确 地 
| 获取 SQL 语句 在 执行 过 程 中 出 现 的 错误 ， 如 果 没有 错误 则 不 输出 任何 内 容 。 

| 1.， 错误 举例 

| 首先 建立 一 个 数据 库 “test”， 在 该 数据 库 下 建立 一 个 数据 表 “user”。 该 数据 表 的 结构 如 
| 表 4.1 所 示 。 
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表 4.1 数据 表 user 结构 | 

字 段 名 类 型 . 说 有明 | 

id Int(10) 自动 编号 | 

user 用 户 名 | 

pass 密码 | 
lxdh 联系 电话 
lxdz 联系 地 址 


然后 建立 与 数据 库 的 连接 ，connectphp 的 代码 如 下 : 


<2php 

$id=mysql connect(localhost,root,root); 
mysql_select_db('test',$id ); 

ye 


连接 数据 库 成 功 后 将 建立 一 个 表单 提交 的 页 面 submit.php， 代 码 如 下 : 


<html><head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 
<title> 错 误 调试 </title></head> 
<body> 
<table width="300" border="1" align="center" cellpadding="0" cellspacing="0"> 
<form name="form1" method="post" action="submit_ok.php"> 
<tr> 
<td width="94" height="30" align="center" valign="middle"> 用 户 名 :</td> 
<td width="200"><input name="user" type="text" id="user" size="20"></td> </tr> 
<tr> 
<td height="31" align="center" valign="middle"> 密 码 :</td> 
<td><input name="pass" type="text" id="pass" size="20"></td> </tr> 
<tr> 
<td height="28" align="center" valign="middle"> 联 系 电 话 :</td> 
<td><input name="lxdh" type="text" id="lxdh" size="20"></td> </tr> 
<tr> 
<td height="30" align="center" valign="middle"> 联 系 地 址 :</td> 
<td><input name="Ixdz" type="text" id="lxdz" size="20"></td></tr> 
<tr> | 
<td height="30" colspan="2" align="center" valign="middle"><input name="submit" type="submit" | 
id="submit" value=" 提 交 "></td> </tr> | 
</form></table> 
<body></html> 


进行 数据 处 理 的 文件 是 submit.php， 代 码 如 下 : 


<?php 
include("connect.php"): // 连 接 到 数据 库 
if($submit==" 提 交 "){ 
$query="insert into user (user,pass,Ixdh,lxdz) values('$user','$pass','$lxdh','$lxdz")"; 
$result-mysql query($query.$id); 
if($resul) { 
echo "<meta http-equiv=\"Refresh\" content=\"2:;url=$furl\">"; 
echo "发 布 成 功 !"; 
}else{ echo "发 布 主题 失败 !1"; } 
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通过 上 述 3 个 文件 ， 使 用 MySQL 语句 的 错误 基本 就 可 以 展示 出 来 了 。 
会 内 | 第 一 个 方面 的 错误 是 数据 提交 表单 与 数据 处 理 之 间 的 错误 ， 即 在 表单 提交 页 文本 域 中 输入 
一 | 的 名 称 与 数据 处 理 页 中 使 用 的 变量 不 一 致 。 笔 者 通过 一 个 组 合 的 图 片 来 展示 这 个 错误 , 如 图 4.9 


| 从 图 4.9 中 可 以 看 到 ， 在 设置 用 户 名 的 文本 字段 时 ， 文 本 域 中 填写 的 是 “usee”， 而 在 数据 
| 处 理 页 中 ， 通 过 提交 使 用 的 变量 则 是 “$user”， 这 就 是 错误 所 在 。 这 个 错误 在 程序 运行 的 过 程 
| 中 是 看 不 出 来 的 ， 程 序 仍然 正常 运行 ， 但 是 当 进入 到 数据 库 中 查看 数据 时 就 会 出 现 如 图 4.10 


Ti user pass lxdh Iau 


口 轩 和 2 "R123456，7785792 “再 林 " 


口 可 目 3 " 汉 将 ” “450789” 5202588 “松原 ” 
口 可 二 !“ 源 去 ” “123456”7680305 “长 春 " 
口 轩 是 4 "就 守 ”7456129” 1335031 “四 于 
DHes "123456” 7680305 “白城 ” 
图 4.9 错误 示例 (一 ) 图 4.10 数据 表 中 的 数据 


| 其 中 “id” 为 6 的 记录 中 的 “user” 字 段 的 值 是 空 的 ， 其 原因 就 是 由 于 文本 域名 称 和 提交 
| 的 变量 不 一 致 。 

| 第 二 个 方面 的 错误 是 在 数据 处 理 页 中 , 在 
| 该 页 的 SQL 语句 中 字段 名 称 排列 的 顺序 与 提 机 
| 交 的 变量 值 的 顺序 不 一 致 ， 如 图 4.11 所 示 。 ,ee Rei 
| 如 图 4.11 所 示 , 在 insert 语 句 中 的 字段 的 ” 呈 
， 顺 序 是 “userpass,lxdh,lxdz”, 而 字段 的 值 的 顺 图 4.11 错误 示例 (二 ) 

| 序 是 “'$user,$lxdh',,$pass,$lxdz”， 其 中 颠倒 了 “lxdh” 和 “pass” 的 顺序 ， 该 程序 仍然 可 以 继 
| 续 运 行 ， 但 是 在 查看 数据 库 时 问题 就 显示 出 来 了 ， 如 图 4.12 所 示 。 

| 从 数据 库 中 可 以 看 到 ， 第 7 条 记录 的 “pass” 和 “jlxdh” 字 段 的 值 被 颠倒 了 ,“lxdh” 字 段 
| 的 值 被 添加 到 “pass” 中 了 ， 另 外 ， 由 于 “lxdh” 字 段 的 类 型 是 “int”， 不 支持 “varchar” 类 型 
| 的 值 ， 导 致 “lxdh” 字 段 中 没有 值 。 

| 第 三 个 方面 的 错误 是 insert 语句 中 的 字段 名 和 数据 库 中 的 字段 名 称 不 一 致 ， 如 图 4.13 所 示 。 
| 其 中 的 字段 “lxsh” 和 数据 表 中 的 字段 名 称 “lxdh” 不 一 致 ， 这 个 错误 直接 导致 数据 提交 
| 失败 。 
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| 图 4.12 图 4.13 错误 示例 (三 ) 
J .144 。 


2. 获取 错误 信息 
使 用 “echo mysql] error():” 
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来 输出 错误 信息 ， 该 语句 被 放置 在 数据 处 理 页 中 使 用 ， 有 具体 的 


位 置 是 在 mysql_query0 函 数 输 4 


通过 该 语句 的 使 用 输出 错误 


8 的 后 面 ， 如 图 4.14 所 示 。 
信息 ， 如 图 4.15 所 示 。 


《3?php 
include (connect Ve 
ibmit==" 提 交 ”) 


必 http://127.0.0.1/ar/f1/... 属 ] 固 | 攻 ] 
文件 中 转 钼 @) 查看 山 收 头 W ” 剧 


地 让 国 ) | 罩 http:/1127.0.0. lnrya/l 同 加 和 


输出 错误 信息 语 避 
et 二 httpret te Refresh\” content=\"2;url=$furl\”. 
a “ 训 据 提交 成 功 
lelsel e ho " 贡 所 提交 失 耻 11” } 


Inknowmn colum "lxsh’ in ’field 


beg 数据 提交 失败 以 


| 
| 
| 
OB Internet | 
| 
| 
| 
| 


图 4.15 输出 错误 信息 的 页 面 


图 4.14 输出 错误 信息 语句 的 使 用 


通过 mysql_error() 函 数 获取 错误 信息 ， 不 但 可 以 获取 insert 添加 语句 的 错误 信息 ， 而 且 也 | 
适用 于 select、update 和 delete 语句 ， 同 时 还 可 以 在 连接 数据 库 服务 器 及 选择 数据 库 时 应 用 此 | 
函数 ， 获 取 连 接 、 选 择 失败 时 的 错误 信息 。 例 如 : | 
<2php | 
$id=mysql_ connect(localhost,root,111") or die(mysql_errorO): 
mysql_select_db('test',$id ) or die(mysql_errorO); 
mysql_query("set names gb2312"); 
2> 


问题 108 ”如 何 执 行 数 据 的 批量 添加 ? 


问题 阐述 
如 何 执行 数据 的 批量 添加 ? 
专家 解答 | 


数据 的 批量 添加 使 用 LOAD DATA 和 MySQLIMPORT 语句 实现 。 
LOAD DATA 语句 ， 通 过 读 取 本 地 系统 上 的 文件 ， 可 以 将 大 量 数据 添加 到 数据 库 中 。 语 法 
如 下 : | 
load data local infile "filename.txt" into table table_name | 


其 中 ，filename txt 是 当前 目录 中 的 数据 文件 的 名 称 ，table_name 是 要 被 装载 的 数据 表 。 ”| 
LOAD DATA 语句 读 取 客 户 机 当前 目录 中 数据 文件 fiename.txt 的 内 容 , 将 其 内 容 装载 到 数 
据 表 table_ name 中 。 | 
MySQLIMPORT 语句 ， 实 用 程序 直接 从 文件 读 取 批量 数据 ， 它 相当 于 load data 语句 的 一 | 

个 接口 。 语 法 如 下 : | 
%MySQLimport -local table_ name filename .txt : | 
MySQLIMPORT 可 以 自动 生成 一 个 LOAD DATA 语句 ， 该 语句 把 fename.txt 文件 中 的 数 
.145 。 


Pip 六 预 知道 的 300 个 问题 


据 装 入 table_ name 表 中 。MySQLIMPORT 根据 文件 名 导出 表 名 ， 即 将 文件 名 第 一 个 圆 点 前 的 
所 有 字符 作为 表 名 。 例 如 ， 文 件 class.txt 被 装 入 class 表 中 。 


问题 109 ”如 何 规范 地 定义 数据 表 中 的 字段 名 称 ? 


问题 讲述 


数据 表 中 的 字段 名 称 用 来 代表 一 组 数据 的 名 称 , 合理 地 为 每 组 数据 定义 最 贴切 的 字段 名 称 


是 很 有 必要 的 ， 那 么 ， 如 何 规范 地 定义 数据 表 中 的 字段 名 称 呢 ? 
| 专家 解答 


数据 表 字 段 的 命名 必须 遵循 以 下 规范 : 
(1) 采用 有 意义 的 字段 名 。 字 段 的 名 称 必 须 是 易于 理解 ， 能 表达 字段 功能 的 英文 单词 或 


缩写 英文 单词 ， 单 词 首 字母 必须 大 写 ， 一 般 不 超过 3 个 英文 单词 。 例 如 ， 人 员 信 息 表 中 的 电话 
| 号 码 可 命名 为 : Telephone 或 Tel。 产 品 明细 表 中 的 产品 名 称 可 用 ProductName 表示 推荐 一 般 


| 用 完整 的 英文 单词 )。 


(2) 系统 中 所 有 属于 内 码 字 段 ( 仅 用 于 标示 唯一 性 和 程序 内 部 用 到 的 标示 性 字段 )， 名 称 


取 为 :“ID”， 采 用 整 型 或 长 整 型 数 ， 具 体 根据 可 能 的 数据 量 确定 ， 增 加 记录 时 取 最 大 值 加 1， 
| 该 字段 通常 为 主 关键 字 。 


(3) 在 命名 表 的 列 时 不 要 重复 表 的 名 称 。 例 如 ， 在 名 为 Employee 的 表 中 避免 使 用 名 为 


| EmployeeLastName 的 字段。 


(4) 不 要 在 列 的 名 称 中 包含 数据 类 型 。 
(5) 字段 最 好 填写 描述 信息 。 
(6) 在 定义 数据 表 中 字段 的 名 称 时 ， 切 记 不 能 使 用 特殊 的 关键 字 ， 如 mysql、time、 


| datetime、database 等 。 


例如 ， 下 面 创建 tb_book 表 ， 并 创建 相应 字段 ， 如 图 4.16 所 示 。 
专家 点 评 
在 创建 字段 时 ， 最 好 为 字段 添加 注释 ， 这 样 可 以 方便 程序 维护 及 调试 。 


四 服务 器 : 192.168.1189 加 教 据 库 : db_book 》 国 狼 :tb_book 
周 训 鉴 ， 辐 兰 构 “只 soL。 己 搜 索 了 插入 及 导出 上 腾 mpor， 移 欣 作 症 洁 字 加 开除 


有 类 型 整理 。。 尾 性 Nil 到 这 往外 扣 作 
Fu nttio) 再 auto_increment 弹 六 本 加 加 
sort varchar(100) ue_unicode_ci 理 EE 
ams varchartl00) 。 UB_unicode 天 加 加 区 加 加 
Py ee varchar(l00) ure_unicode_ 理 已 X 国 四 加 国 
厂 MBs] achartl0000) |] 否 习 必 xx 久 加 如 晤 
emiakog orchar(l0000) a 否 司 殉 网 团 加 右面 
3 oi 否 Ei: 
一 El 下 Eo 
| Whoa i 区 ED 
ae 本 习 必 X 国 可 加 本 


人 全 大 /全 让 选 兴 姑 呈 国 X 国 四 如 即 


4.16 创建 数据 表 字 段 
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4 p> 
第 生 章 数据 库 操作 9 3 | 


问题 110 ”如何 查询 指定 时 间 段 的 数据 ? 


问题 阐述 
在 查询 数据 库 中 的 数据 时 ， 如 何 实现 查询 指定 时 间 段 的 数据 呢 ? 
专家 解答 
1. 方法 分 析 
在 SQL 语句 中 实现 对 指定 范围 内 的 数据 进行 查询 可 以 通过 以 下 两 种 方式 实现 ， 第 一 种 方 
式 是 通过 关键 字 between...and... 实 现 ， 语 法 格式 如 下 : 
select 要 查找 的 字段 from 表 名 where 字段 名 between 初始 值 and 终止 值 
第 二 种 方式 是 通过 比较 运算 符 实现 ， 语 法 格式 如 下 : 
select 要 查找 的 字段 from 表 名 where 字段 名 > 初始 值 and 字段 名 < 终止 值 
2. 应 用 示例 
在 图 书 管理 系统 中 ， 查 询 指定 出 版 日 期 范围 内 的 图 书 情况 。 其 关键 步骤 如 下 : 
创建 index.php 文件 ， 启 动 MySQL 服务 并 连接 MySQL 数据 库 。 当 单 击 “ 查 看 ”按钮 时 ， 
显示 指定 时 间 段 的 数据 。 代 码 如 下 : 
<?php 
这 $_POST[sub]){ // 单 击 查看 按钮 
// 判 断 文 本 框 信息 
这 $_ POST[textl] 一 "" | $_POST[text1] 一 "输入 开始 日 期 " || $_POST[text] 二 "" 
$_POST[text] 一 "输入 结束 日 期 "){ 
echo "<script>alert(' 请 输入 日 期 );</script>";， ”//JavaScript 提示 
}else{ 
// 验 证 日 期 格式 的 正则 表达 式 


iftpreg_match("/([0-9]{3}[1-9][0-9]{2}[1-9][0-9]{1}[1-9][0-9]{2}I[1-9][0-9]{3})-(CO[13578]l1[0-2D)- 
Clo he ee ll 
// 连 接 MySQL 


$conn = mysql_ connect("localhost","root","111") or die("connect mysql false"); 
// 连 接 MySQL 数据 库 
Inysql _select db("db_ database",$conn) or die("connect database false"):; 
mysql_query("set names utf8"); /设置 编码 格式 
$rss = mysql query("select * from tb demo047 where date between 
'$_POST[textl] and '$_ POST[textl' order by date"): /返回 结果 集 
echo'<table width="S$80px"><tr><td background="pic/head.JPG">ID</td><td 
background="pic/head.JPG"> 书 名 </td><td ”background="pic/head.JPG"> 价 格 </td><td ” background= 
"pic/head.JPG"> 日 期 <td></tr>'; 
while($rst = mysql fetch array($rss){ // 循 环 输出 结果 
> 
<tr><td background="pic/headJPG"><?php echo S$rst[0]:?></td><td background= 
"pic/head.JPG"><?php echo S$rst[1]:?></td><td background="pic/head.JPG"><?php echo S$rst[2]:?></td><td 
background="pic/head.JPG"><?php echo $rst[3]:?></td></tr> 
<?php 
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| } 
| Jelse{ 
| //JavaScript 提示 信息 
| echo "<script>alert( 日 期 格式 错误 ! 正确 格式 为 : xxxx-xx-xx');</script>"; 
| } 
天 |] ) 
| } 
| 运行 本 示例 ， 如 图 4.17 所 示 ， 首 先 在 图 中 的 文本 框 中 输入 日 期 范围 ， 然 后 单 击 “ 查 看 ” 按 


| 钮 即 可 实现 查找 该 日 期 范围 内 的 所 有 图 书信 息 。 


DEEP 


[EL 


局 日 期 

PHP24 堂 课 2011-01-11 
5824 堂 课 元 2011-01-17 
JAVA24 语 麻 2011-01-17 
524 党 课 2011-01-17 


图 4.17 查询 指定 时 间 段 的 数据 
问题 111 如 何 按 月 查询 统计 数据 ? 


| 问题 请 述 


在 查询 数据 库 中 的 数据 时 ， 如 何 按 月 查询 统计 数据 呢 ? 


| 专家 解答 


SQL 语言 中 提供 了 如 下 函数 ， 利 用 这 些 函数 可 以 很 方便 地 实现 按 年 、 月 、 日 进行 查询 。 
year(data): 该 函数 用 于 返回 日 期 表达 式 data 中 的 公 ws 
month(data): 该 函数 用 于 返回 日 期 表达 式 data 中 的 月 份 所 对 应 的 数值 。 
day(data): 该 函数 用 于 返回 日 期 表达 式 data 中 的 日 期 所 对 应 的 数值 。 
例如 ， 按 月 查询 图 书信 息 ， 其 SQL 语句 如 下 : 
select * from tb _demo where month(date) ='$_POST[textl]' order by date 


问题 112 ”如 何 实现 在 删除 数据 前 给 予 提示 信息 ? 


| 问题 阐述 

| 在 开发 网 站 的 过 程 中 , 为 了 避免 将 重要 的 数据 删除 ， 有 必要 在 执行 删除 操作 之 前 提醒 一 下 
| 管理 者 ， 是 否 要 删除 该 数据 ， 这 样 可 以 避免 误 删 数据 。 那 么 ， 如 何 实现 在 删除 数据 前 给 予 提示 
| 信息 呢 ? 
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专家 解答 
1. 解决 思路 
删除 数据 前 给 予 提示 信息 ， 应 用 JavaScript 脚本 实现 ， 其 代码 如 下 : 
<script language="javascript"> 
function delconfirm(x){ 


if(window.confirm(' 您 确定 删除 该 项 吗 ?")=true){ 
window.location.href="index_ok.php?ljid="+x;}} | 
</script> 


使 用 window 对 象 中 的 confirm() 方 法 实现 在 确认 对 话 框 中 显示 指定 的 字符 串 ， 并 通过 
window 对 象 的 location 属性 指定 当前 文档 要 跳 转 到 的 
URL。 

2. 应 用 示例 

本 例 实 现在 执行 删除 操作 时 弹出 一 个 提示 对 话 
框 “您 确定 删除 该 项 吗 ?”， 运行 结果 如 图 4.18 所 示 。 

实现 步骤 如 下 : 

(1) 创建 与 数据 库 的 连接 ， 代 码 如 下 : 
<2php 
$id=mysql_connect("localhost","root","root")or dir( 连 接 服 务 器 失败 mysql_error0); 
if(mysql select db("db database",$id)){ 
本 
}else{fecho (连接 数据 库 失 败 :'. mysqL_errorO):} 


mysql_query("set names gb2312"); 
?> 


(2) 从 数据 库 中 读 取 数 据 ， 设 置 删除 操作 的 按钮 ， 编 写 JavaScript 脚本 实现 在 删除 数据 之 
前 弹出 对 话 框 给 予 提示 ， 关 键 代码 如 下 : 


<?php session_start():include("Conn/conn.php"):$furl=getenv("HTTP REFERER")?> | 

<script language="javascript"> 

function delconfirm(x){ 

这 window.confimm(' 您 确定 删除 该 项 吗 ?) 一 tmue){ 
window.location.href='index_ok.php?ljid="+x:;}} 

</script> | 

<?php | 
$query="select * from tb_delete where id limit 4"; | 
$result=mysql_query($query): | 

whbile(@$myrow=mysql fetch_array($resulb){ 

了 > 


图 4.18 删除 数据 前 给 予 提示 信息 


<tr> 

<td width="77" height="36" align="center"><span class="STYLE3"><?php echo $myrow[user]:?> 
</span></td> | 
<td width="96" align="center"><span class="STYLE3"><?php echo $myrow[title]:?></span></td> | 

<td width="155" align="center"><span class="STYLE3"><?php echo $myrowltext]:?></span></td> | 

<td width="95" align="center"><span class="STYLE3"><?php echo $myrow[data]l:?></span></td> ! 

<td width="77" align="left"><input name="Submit" type="submit" onClick="delconfirm(<?php | 

echo $myrow[id]:?>)" value=" 删 除 "></td> | 
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</tr> 
<2php }2> 


(3) 执行 数据 的 删除 操作 ， 代 码 如 下 : 


全 | | <?php session start(:include("Conn/conn.php"):$furl=getenv("HTTP REFERER"); 
NA | if($1jid==true) { 


| Sljid=$_GET[jid']: 
$query="delete from tb_delete where id="$1ljid"™"; 
| Sresult=mysql_query(Squery); 
if($result—true){ 
echo "删除 成 功 !1"; 
echo "<meta http-equiv=\"Refresh\" content=\"2;url=index.php\">"; 
}else{echo "删除 失败 1";}} 


问题 113 ”如 何 查 询 日 期 型 数据 ? 
| 问题 亲 述 
| 如 何 查 询 日 期 型 数据 ? 
| 专家 解答 


| 不 同 的 数据 库 对 日 期 型 数据 的 查询 是 有 区 别 的 , 下面 将 以 几 种 典型 的 数据 库 为 例 讲解 在 不 
| 同 的 数据 库 中 对 日 期 型 数据 的 查询 方式 。 
下 面 均 是 以 在 图 书 表 (tb_demo) 中 查询 出 版 日 期 (date) 为 “2011-07-19” 为 例 进行 讲解 的 。 
1，MySQL 数据 库 中 对 日 期 型 数据 进行 查询 
select * from tb_demo where birthday='2011-07-19' 
2. SQL Server 数据 库 中 对 日 期 型 数据 进行 查询 
Select * from tb_demo where birthday="2011-07-19" 
3，Access 数据 库 中 对 日 期 型 数据 进行 查询 
Select * from tb_demo where birthday=#2011-07-19# 


| 通过 上 面 3 个 例子 可 以 发 现 , 在 MySQL 数据 库 和 SQL Server 数据 库 中 实现 对 日 期 型 数据 
， 查 询 所 要 查询 的 日 期 应 用 单 引号 括 起 来 ， 而 在 Access 数据 库 中 使 用 JET SQL 语法 查询 时 所 查 
| 询 的 日 期 应 用 “#” 号 括 起 来 。 


问题 114 ”如 何 使 用 左 连 接 (left outer join ) 查询 数据 ? 


问题 阐述 


| 如果 要 包含 连接 的 两 个 表 中 的 不 匹配 行 ， 可 以 通过 左 连 接 或 右 连接 的 方式 实现 。 那 么 ， 如 
| 何 使 用 左 连 接 查 询 数据 呢 ? 
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专家 解答 


左 连接 返回 的 查询 结果 包含 左 胡 中 的 所 有 符合 查询 条 件 及 右 表 中 所 有 满足 连接 条 件 的 行 ， 
MySQL 数据 库 中 使 用 左 连接 的 语法 格式 如 下 : 
select field 1[field2...] from tablel left [outer] join table2 on join_condition [where search_condition] 


参数 说 明 : 

@ left outer join: 表示 表 之 间 通 过 左 连接 方式 相互 连接 ， 也 可 以 简写 成 left join。 

@ on join _ condition: 指 多 表 建 立 连 接 所 使 用 的 连接 条 件 。 

@ where search_condition: 可 选项 ， 用 于 设置 查询 条 件 。 

例如 , 查询 员工 信息 表 (tb_demo078_informaction) 中 所 有 记录 以 及 员工 工资 表 (tb_demo078_ 
wages) 中 与 员工 ID 相 匹配 的 记录 。 | 

首先 ， 进 入 phpMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按 钮 ， 进 入 输入 | 
查询 语句 页 面 ， 如 图 4.19 所 示 。 | 

输入 完 查 询 语 句 后 ， 单 击 “ 执 行 ” 和 即 可 显示 查询 结果 ， 如 图 4.20 所 示 。 
Ts 
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图 4.19 在 phpMyAdmin 中 输入 查询 语句 图 4.20 查询 结果 
专家 点 评 


左 连接 是 以 left outer join 左 侧 的 表 名 为 基准 的 ， 故 称 左 连接 ， 左 侧 表 名 的 所 有 信息 将 被 全 
部 输出 ， 符 合 条 件 的 右 侧 表 信 息 也 会 被 输出 ， 如 果 不 符 合 条 件 ， 则 返回 为 NULL。 | 


问题 115 ”如 何 使 用 右 连 接 (right outer join ) 查询 数据 ? 


问题 阐述 | 
通过 右 连 接 可 以 将 右 表 中 满足 查询 条 件 的 记录 全 部 显示 出 来 并 显示 左 表 中 满足 连接 条 件 
的 记录 。 那 么 ， 如 何 使 用 右 连 接 (right outer join) 查询 数据 呢 ? | 
专家 解答 
右 连接 返回 的 查询 结果 包含 左 表 中 的 所 有 符合 连接 条 件 以 及 右 表 中 所 有 满足 查询 条 件 的 | 
行 ，MySQL 数据 库 中 使 用 右 连接 的 语法 格式 如 下 : | 
select field 1[field2...] from tablel right [outer] join table2 on join_condition [where search_condition] 
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参数 说 明 : 

@ right outer join: 表示 表 之 间 通 过 右 连 接 方式 相互 连接 ， 也 可 以 简写 成 right joim。 

@ onjoin_condition: 指 多 表 建 立 连接 所 使 用 的 连接 条 件 。 

@ where search condition: 可 选项 ， 用 于 设置 查询 条 件 。 

例如 ， 将 员工 信息 表 中 满足 连接 条 件 的 记录 及 员工 工资 表 中 所 有 记录 全 部 显示 出 来 。 

首先 ， 进 入 phpMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按 钮 ， 进 入 输入 
， 查 询 语句 页 面 ， 如 图 4.21 所 示 。 

输入 完 查 询 语句 后 ， 单 击 “ 执 行 ”按钮 ， 即 可 显示 查询 结果 ， 如 图 4.22 所 示 。 
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图 4.21 在 phpMyAdmin 中 输入 查询 语句 图 4.22 查询 结果 
专家 点 评 


右 连接 是 以 right outer join 右 侧 的 表 名 为 基准 的 ， 故 称 右 连接 ， 右 侧 表 名 的 所 有 信息 将 被 
全 部 输出 ， 符 合 条 件 的 左 侧 表 信息 也 会 被 输出 ， 如 果 不 符合 条 件 ， 则 返回 为 NULL。 


问题 116 ”如 何 对 统计 结果 进行 排序 ? 


问题 阐述 
在 商品 信息 管理 系统 中 ， 经 常 需要 对 商品 的 价格 、 数 量 进行 求 和 ， 然 后 根据 所 求 得 的 值 对 
商品 进行 排序 ， 从 而 实现 商品 的 排序 输出 。 那 么 ， 如 何 对 统计 结果 进行 排序 呢 ? 
专家 解答 
使 用 ORDER BY 可 以 对 查询 的 结果 进行 升序 (ASC) 和 降序 (DESC) 排列 , 在 默认 情况 下 ， 
ORDER BY 按 升序 输出 结果 。 如 果 要 按 降序 排列 ， 则 可 以 使 用 DESC 来 实现 。 语 法 格式 如 下 : 
ORDER BY 字段 名 [ASCIDESC]; 


ASC 表示 按 升 序 进 行 排序 。 
DESC 表示 按 降序 进行 排序 。 


说 明 : 
当 对 含有 NULL 值 的 列 进行 排序 时 ， 如 果 是 按 升序 排列 ， 则 NULL 值 将 出 现在 最 前 面 ; 
如 果 是 按 降序 排列 ， 则 NULL 值 将 出 现在 最 后 。 
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的 所 有 信息 ， 按 照 “id” 进 行 降序 排列 ， 查 询 语句 如 下 : 


例如 ， 查 询 tb login 表 叶 
select * from tb login order by id desc; 


查询 结果 如 图 4.23 所 示 。 


图 4.23 按 id 序 号 进行 降序 排列 


问题 117 如 何 使 用 聚集 函数 SUM 对 学 生成 绩 进 行 汇 总 ? 


问题 阐述 
在 学 生成 绩 管理 系统 中 ,经 常 需要 求 出 班级 所 有 学 生 的 各 科 成 绩 总 和 ， 从 而 求 出 所 有 学 生 
的 各 科 平均 成 绩 。 那 么 ， 如 何 使 用 聚集 函数 SUM 对 学 生成 绩 进行 汇总 呢 ? | 
专家 解答 
在 SQL 语句 中 ， 实 现 对 某 字段 的 所 有 记录 进行 求 和 应 用 的 是 sum() 函 数 ， 其 语法 如 下 : 


sum([all | distinct] expression) 


参数 说 明 : | 
@ all， 表示 对 指定 字段 的 所 有 值 进行 聚集 函数 运算 ，all 为 默认 值 ， 如 果 不 加 参数 al 或 | 


distinct 则 表示 对 指定 字段 的 所 有 记录 进 和 
@ distinct: 表示 对 指定 字段 的 所 有 非 重 复 记录 进行 求 和 。 
日 expression: 是 精确 数字 或 近似 数字 数据 类 型 分 类 bit 数据 类 型 除外 ) 的 表达 式 。 
sum() 函 数 的 返回 类 型 如 表 4.2 所 示 。 
表 4.2 sum() 函 数 的 返回 类 型 


表达 式 结果 返回 类 型 
整数 分 类 int 
decimal 分 类 decimal(38,s) 除 以 decimal(10,0) 
money 和 smallmoney 分 类 mone. 
float 和 real 分 类 float 


例如 ， 下 面 将 学 生 的 各 科 成 绩 进行 汇总 ， 得 出 总 分 。 | 
首先 ， 进 入 phpMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按 钮 ， 进 入 输入 | 


查询 语句 页 面 ， 如 图 4.24 所 示 。 | 
输入 完 查 询 语句 后 ， 单 击 “ 执 行 ”按钮 ， 即 可 显示 查询 结果 ， 如 图 4.25 所 示 。 | 
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图 4.24 在 phpMyAdmin 中 输入 查询 语句 
专家 点 评 


sum() 函 数 主要 被 用 于 数据 表 字段 求 和 ， 
查询 之 中 才能 真正 体现 出 它 的 优势 。 


图 4.25 查询 结果 


一 般 与 分 组 函数 联合 使 用 ， 但 是 只 有 出 现在 多 表 


问题 118 ”如 何 使 用 聚集 函数 MIN 求 利 润 最 少 的 商品 ? 


问题 前 述 


在 销售 管理 系统 中 ， 商 家 会 根据 不 同 商品 的 销售 利润 来 确定 该 商品 的 进货 数量 。 


何 使 用 聚集 函数 MIN 求 利润 最 少 的 商品 呢 ? 
专家 解答 


SQL 语言 中 获取 某 字 段 的 最 小 值 应 用 min0 函 数 ， 函 数 的 语法 如 下 : 


min([all | distinct] expression) 


例如 ， 查 询 出 销售 利润 最 小 的 商品 信息 。 

首先 ， 进 入 phppMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按 钮 ， 进 入 输入 
查询 语句 页 面 ， 如 图 4.26 所 示 。 

输入 完 查 询 语句 后 ， 单 击 “ 执 行 ” 按 钮 ， 
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图 4.26 在 phpMyAdmin 中 输入 查询 语句 


图 4.27 查询 结果 
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问题 119 如何 使 用 聚集 函数 MAX 求 销 售 
利润 最 高 的 商品 ? 


问题 阐述 

在 商品 销售 管理 系统 中 ， 商 品 的 销售 利润 统计 模块 是 必 不 可 少 的 。 那 么 ， 如 何 使 用 聚集 函 
数 MAX 求 销售 利润 最 高 的 商品 呢 ? | 
专家 解答 

在 SQL 语言 中 ， 应 用 max0 函 数 获取 字段 中 的 最 大 值 ， 函 数 的 语法 如 下 : 

max([all | distinct] expression) 

参数 说 明 ; | 

@ all: 表示 对 指定 字段 的 所 有 值 进 行 聚集 函数 运算 ，all 为 默认 值 ， 如 果 不 加 参数 all 或 | 
distinct 则 表示 对 指定 字段 的 所 有 记录 进行 聚集 运算 。 | 

@ distinct: 表示 对 指定 字段 的 所 有 非 重 复 记录 进行 求 最 大 值 。 

日 expression: 是 精确 数字 或 近似 数字 数据 类 型 分 类 (bit 数据 类 型 除外 ) 的 表达 式 。 

例如 ， 查 询 2005 年 销售 额 最 多 的 商品 信息 。 | 

首先 ， 进 入 phpMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按 钮 ， 进 入 输入 | 
查询 语 名 页面， 如 图 4.28 所 示 。 | 

输入 完 查 询 语句 后 ， 单 击 “ 执 行 ”按钮 ， 即 可 显示 查询 结果 ， 如 图 4.29 所 示 。 
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图 4.28 在 phpMyAdmin 中 输入 查询 语句 图 4.29 查询 结果 


问题 120 ”如何 使 用 INSERT(s1,x,len,s2) 函 数 
替换 指定 字符 ? 


问题 讲述 


在 MySQL 数据 库 中 有 可 能 会 遇 到 这 样 一 种 情况 ， 在 海量 的 数据 记录 中 ， 存在 一 个 错误 的 | 
字符 串 ， 需 要 对 这 个 错误 的 字符 串 进 行 修改 ， 采 用 什么 样 的 方法 才 是 最 快捷 、 最 有 效 的 呢 ? 


区 UD 


人 ip 六 预 知道 的 300 个 问题 


解决 此 问题 ， 最 理想 的 方法 就 是 应 用 MySQL 中 的 字符 串 函数 INSERTO， 实 现 对 指定 字段 


| 中 的 字符 串 进行 替换 操作 。 

| INSERT(s1.x.len,s2) 函 数 可 以 将 字符 串 s1 中 x 位 置 开始 长 度 为 len 的 字符 串 蔡 换 为 s2 字符 串 。 

例如 , 使 用 INSERT 函数 将 tb_book 表 中 marker 字段 中 的 第 3 个 字符 开始 的 两 个 字符 替换 
”为 book 字符 。 

| 首先 ， 进 入 phpMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按 钮 ， 进 入 输入 

| 查询 语 名 页面， 如 图 4.30 所 示 。 

| 输入 完 查询 语句 后 ， 单 击 “ 执 行 ”按钮 ， 即 可 显示 查询 结果 ， 如 图 4.31 所 示 。 
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图 4.30 在 phpMyAdmin 中 输入 查询 语句 图 4.31 查询 结果 


| 专家 点 评 
当 数 据 库 中 的 特定 数据 需要 进行 批量 更 新 时 ， 使 用 INSERTO 函 数 可 以 很 好 地 完成 操作 。 


问题 121 如 何 使 用 UPPER(S)、UCASE(s) 函 数 
将 字母 变 成 大 写 ? 


| 问题 阐述 
| 如 何 使 用 UPPER(s)、UCASE(s) 函 数 将 字母 变 成 大 写 呢 ? 
| 专家 解答 


通过 MySQL 中 的 UPPER(s) 函 数 和 UCASE(s) 函 数 可 以 将 字母 变换 成 大 写 

例如 ， 下 面 将 由 _demo019 表 中 name 字段 中 的 英文 全 部 变 成 大 写 
| 首先 ， 进 入 phpMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按 钮 ， 进 入 输入 
| 查询 语 名 页面， 如 图 4.32 所 示 。 
| 输入 完 查询 语句 后 ， 单 击 “ 执 行 ”按钮 ， 即 可 显示 查询 结果 ， 如 图 4.33 所 示 。 


， 专 家 点 评 
| 通过 lower(s) 和 lcase(s) 函 数 可 以 将 大 写字 母 转换 为 小 写 。 
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图 4.32 在 phpMyAdmin 中 输入 查询 语句 图 4.33 ”查询 结果 


问题 122 ”如 何 利 用 CSS 样式 实现 分 页 打印 ? 


问题 阐述 | 
在 制作 数据 打印 程序 时 ， 对 于 多 数据 ( 指 的 是 一 页 纸 不 能 全 部 打印 完毕 的 数据 ) 通常 采用 | 
分 页 打印 。 那 么 ， 如 何 利用 CSS 样式 实现 分 页 打印 呢 ? 


专家 解答 


利用 CSS 样式 实现 分 页 打印 ， 其 主要 应 用 thead 标记 、tfoot 标记 和 page-break-after 属性 。 | 
(1) thead 标记 | 
thead 用 于 设置 表格 的 表 头 。 
(2) tfoot 标记 
tfoot 用 于 设置 表格 的 表 尾 。 
(3) page-break-after 属性 | 
page-break-after 属性 在 打印 文档 时 发 生 作用 ， 用 于 进行 分 页 打印 ， 但 是 对 于 <br> 和 <hr> 对 | 
象 不 起 作用 。 其 语法 格式 如 下 : | 
page-break-after: auto | always | avoid | left | right | null 


参数 说 明 : 

@ page-break: 打印 时 在 样式 控制 的 对 象 前 后 换 页 。 

@ after: 设置 对 象 后 出 现 页 分 隔 符 。 设 置 为 always 时 ， 始 终 在 对 象 之 后 插入 页 分 隔 符 。 
@ auto: 需要 在 对 象 之 后 插入 页 分 隔 符 时 插入 。 

@ always: 始终 在 对 象 之 后 插入 页 分 隔 符 。 

@ avoid: 未 支持 。 避 免 在 对 象 后 面 插入 分 隔 符 。 

@ left: 未 支持 。 在 对 象 后 面 插入 页 分 隔 符 ， 直 到 它 到 达 一 个 空白 的 左 页 边 。 
@ right: 未 支持 。 在 对 象 后 面 插入 页 分 隔 符 ， 直 到 它 到 达 一 个 空白 的 右 页 边 。 
@null: 空白 字符 串 。 取 消 了 分 隔 符 设 置 。 

应 用 示例 

利用 CSS 样式 实现 分 页 打印 。 其 具体 步骤 如 下 : 
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加 
Ct 
(1) 编写 用 于 控制 指定 内 容 不 打印 的 CSS 样式 ， 代 码 如 下 : 


@media print{ 
.bgnoprint{ 
background:display:none; 
} 
-noprint{ 
display:none 
} 


} 


(2) 应 用 include 命令 连接 数据 源 文件 ， 并 应 用 do.…while 循环 语句 输出 图 书信 息 到 浏览 
器 ， 同 时 设置 好 表 头 、 表 尾 及 打印 分 页 ， 关 键 代码 如 下 : 


<2php include "conn/conn.php"; ?> 
<table width="99%" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td height="27" align="center" style=" font-size:14px:"><b> 图 书信 息 查询 </b></td> 
</tr> 
</table> 
<table width="98%" border="0" cellpadding="0" cellspacing="1" bgcolor="#000000" bordercolor= 
"#FFFFFF" bordercolordark="#000000" bordercolorlight="#FFFFFF" > 
<thead style="display:table-header-group:"> <!-- 设 置 表 头 --> 
<tr bgcolor="#EFEFEF"> 
<td width="6%" height="20" align="center"> 编 号 </td> 
<td width="27%" align="center"> 图 书 名 称 </td> 
<td width="23%" align="center"> 内 容 简 介 </td> 
<td width="8%" align="center"> 定 价 </td> 
<td width="10%" align="center"> 作 者 </td> 
<td width="15%" align="center"> 出 版 社 </td> 
<td width="11%" align="center"> 发 行 时 间 </td> 
</tr> 
</thead> 
<!-- 控 制 分 页 --> 
<?php 
$sql=mysql_query("select * from tb_book"); 
S$info=mysql fetch_array($sq]); 
Srow=1; 
do{ 
?> 
<tr align="center" <?php if($row=—2){ ?>style="page-break-after:always"<?php } ?>> 
<td bgcolor="#EFFFFF"><?php echo $info[id]:?></td> 
<td height="25" align="left" bgcolor="#FFFFFF">&nbsp:<?php echo $info[bookname]:?></td> 
<td align="left" bgcolor="#FFFFFF">&nbsp:<?php echo $info[synopsis]:?></td> 
<td bgcolor="#EFFFFF"><?php echo $info[price]:?></td> 
<td bgcolor="#EFFFFF"><?php echo $info[maker];?></td> 
<td bgcolor="#EFFFFF"><?php echo $info[publisher];?></td> 
<td bgcolor="#FFFFFF"><?php echo $info[issuDate]:?></td> 
</tr> 
<?php 
S$row++; 
}while($info=mysql fetch array($sqD) 


> 


<1- 设 置 表 尾 -> 
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<tfoot style="display:table-footer-group; bordernone:"><tr><td></td></tr></ttoot> 
</table> 


(3) 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 ， 代 码 如 下 : 


‘<object id="Wb" classid="CISID:8856F961-340A-11D0-A96B-00C04Fd705A2" width="0" height="0"> | 
</object> | 


(4) 建立 相关 的 打印 超 链 接 ,， 并 调用 WebBrowser 控件 的 相应 参数 实现 打印 预览 及 打印 功 
能 ， 代 码 如 下 : 

<table width="99%" height="25" border="0" ce ing="0" cellspacing="0"> 

<tr> 
<td height="25" align="right" class="noprint"> 

<a href="#" onClick="document.all.Wb.Execwb(7.1)"> 打 印 预 览 </a> 

<a href="#" onClick="document.all.Wb.Execwb(6,1)"> 打 印 <a> 

</td> 

</t> 
</table> 


运行 本 示例 ， 如 图 4.34 所 示 ， 单 击 “ 打 印 预览 ” 超 链接 ， 可 以 查看 打印 结果 ， 如 图 4.35 
所 示 ; 单 击 “ 打 印 ” 超 链接 即 可 进行 分 页 打印 。 | 
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图 4.35 打印 预览 效果 


问题 123 ”如 何 使 用 RTRIM(s) 函 数 去 掉 字 符 串 
结尾 处 的 室 格 ? 


问题 阐述 
如 何 使 用 RIRIM(s) 函 数 去 掉 字 符 串 结尾 处 的 空格 呢 ? 
。159 。 
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RTRIM(S) 函 数 主要 用 于 去 掉 字 符 串 结尾 处 的 空格 。 
入 | 例如， 下 面 将 耻 _demo019 表 中 name 字段 值 结尾 处 的 室 格 去 掉 。 
全 | ， 首先 ， 进入 phpMyAdmin 中 ， 选 择 db_database08 数据 库 ， 单 击 “SQL” 按钮， 进入 输入 


查询 语句 页 面 ， 如 图 4.36 所 示 。 
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图 4.36 在 phpMyAdmin 中 输入 查询 语句 
| 输入 完 查 询 语句 后 ， 单 击 “ 执 行 ”按钮 ， 即 可 显示 查询 结果 ， 去 除 尾 空格 前 后 如 图 4.37 
| 和 图 4.38 所 示 。 
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图 4.37 去 除 尾 空格 前 图 4.38 ”去除 尾 空格 后 


问题 124 MySQL 数据 库 的 备份 和 恢复 


| 问题 曾 述 

| 数据 库 越 大 ， 数 据 备份 的 问题 就 越 让 人 头疼 。 这 确实 像 是 下 梦 一 般 :如果 服务 器 朋 溃 ， 所 

| 有 东西 都 丢失 了 ， 后 果 是 不 堪 设 想 的 。 那 么 我 们 必须 掌握 最 有 效 的 数据 库 备 份 和 恢复 的 方法 ， 在 

| 灾难 发 生 时 可 以 及 时 进行 补救 ， 那么 什么 样 的 方法 才 是 最 有 效 的 数据 库 备份 和 恢复 的 方法 呢 ? 

| 专家 解答 

| 1 通过 mysqldump 命令 备份 数据 库 

| 在 命令 模式 下 完成 对 数据 的 备份 ， 使 用 的 是 MYSQLDUMP 命令 。 通 过 该 命令 可 以 将 数据 
了 . 160 . 
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以 文本 文件 的 形式 存储 到 指定 的 文件 夹 下 。 
通过 MYSQLDUMP 命令 备份 整个 数据 库 的 操作 步骤 如 下 : 
(1) 选择 “开始 ”菜单 中 的 “运行 ”命令 ， 如 图 4.39 所 示 。 


(2) 在 如 图 4.40 所 示 的 对 话 框 中 输入 “cmd”， 单 击 “ 确 定 ” 按 钮 ， 进 入 命令 模式 。 
EE 
号 本 前 wa 
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图 4.39 选择 运行 命令 图 4.40 进入 命令 模式 


(3) 在 命令 模式 中 直接 输入 “mysqldump -uroot -proot db database16>F:\db database16.txt”， | 
然后 按 回 车 键 即 可 ， 如 图 4.41 所 示 。 | 

其 中 ，-uroot 中 的 “root” 是 用 户 名 ，-proot 中 的 “root” 是 密码 ,“db database16” 是 数据 | 
库 名 ，“F\db_database16.txt” 是 数据 库 备份 存储 的 位 置 。 | 


说 明 : 
在 输入 命令 的 过 程 中 ， 在 “-uroot” 中 是 没有 空格 的 ， 在 该 命令 的 结尾 处 也 没有 任何 的 结 
束 符 ， 只 要 按 回 车 键 即 可 。 


2. 通过 mysql 命令 恢复 数据 库 | 

既然 可 以 对 数据 库 进行 备份 ， 那 么 就 一 定 可 以 对 数据 库 文件 进行 恢复 操作 。 执 行 数 据 库 的 

恢复 操作 使 用 的 是 MySQL 命令 。 其 命令 格式 如 下 : | 
Iysql -uroot -proot db_database <F:\db_database16.txt 


其 中 MySQL 是 使 用 的 全 命令 ，-u 后 的 root 代表 用 户 名 ，-p 后 的 root 代表 密码 ，db_database | 
代表 数据 库 名 (或 表 名 ),“<” 号 后 面 的 “F:\db_databasel6.txt” 是 数据 库 备份 文件 存储 的 位 置 。 | 

下 面 介绍 实现 数据 库 恢复 的 操作 步 又 : 

(1) 选择 “开始 ”菜单 中 的 “运行 ”命令 。 

(2) 在 弹出 的 对 话 框 中 输入 “cmd”， 单 击 “ 确 定 ” 按 钮 ， 进 入 命令 模式 。 | 

(3) 在 命令 模式 中 直接 输入 “mysql -uroot -proot db_database16 <F:vdb databasel6txt”， 然 | 
后 按 回 车 键 即 可 ， 如 图 4.42 所 示 。 


图 4.41 通过 命令 备份 db_database16 数据 库 图 4.42 通过 命令 恢复 db_database16 数据 库 


其 中 ，-uroot 中 的 “root” 是 用 户 名 ， 而 -proot 中 的 “root” 是 密码 ,“db_database16” 是 要 
恢复 的 数据 库 名 ,“F:vdb_database16.txt” 是 数据 库 备份 文件 存储 的 位 置 。 
说 明 : 
在 进行 数据 库 的 恢复 时 ， 在 用 户 的 MySQL 数据 库 中 必须 存在 一 个 空 的 、 将 要 恢复 的 数据 
.161 . 
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库 . 在 MySQL 服务 器 中 创建 一 个 db database16 数据 库 ， 然 后 ， 再 执行 数据 的 恢复 操作 。 
| 专家 点 评 


会 内 | 无 论 作 为 数据 库 管理 人 员 、 维护 人 员 还 是 开发 人 员 , 掌握 如 何 备份 数据 库 都 是 十 分 必要 的 。 
一 一 | 数据 库 备 份 工作 既 可 以 提高 系统 的 安全 性 ， 又 给 管理 者 在 维护 工作 上 带 来 了 极 大 方便 。 所 以 ， 
如 何 备份 MYSQL 数据 库 也 在 MySQL 数据 库 管 理 中 占有 一 席 之 地 。 


问题 125 如 何在 MySQL 中 创建 存储 过 程 ? 
问题 请 


| 自 MySQL 5.0 开始 ，MySQL 就 支持 存储 过 程 。 存 储 过 程 是 一 些 被 用 户 定义 的 SQL 语句 集 
| 合 。 一 个 存储 程序 是 可 以 被 存储 在 服务 器 中 的 一 套 SQL 语句 。 存 储 过 程 可 以 被 程序 、 触 发 器 


| 或 男 一 个 存储 过 程 调用 。 其 具有 执行 速度 快 、 提 高 系统 性 能 、 确 保 数 据 库 安全 等 诸多 优点 。 那 
| 么 如 何在 MySQL 中 创建 存储 过 程 呢 ? 


| 专家 解答 
| 1 存储 过 程 创建 解析 
在 MySQL 中 ， 创 建 存储 过 程 的 基本 形式 如 下 : 


CREATE PROCEDURE sp_name ([proc_parameter[,...]]) 
[characteristic...] routine_body 


| 其 中 sp name 参数 是 存储 过 程 的 名 称 ; proc_parameter 表示 存储 过 程 的 参数 列表 ; 

| characteristic 参数 指定 存储 过 程 的 特性 ，routine_body 参数 是 SQL 代码 的 内 容 。 
| proc_parameter 中 的 参数 由 3 部 分 组 成 ， 它 们 分 别 是 输入 输出 类 型 、 参 数 名 称 和 参数 类 型 。 
| 其 形式 为 [INIOUTIINOUT]param_name type。 其 中 IN 表示 输入 参数 ; OUT 表示 输出 参数 ;INOUT 
| 表示 既 可 以 输入 也 可 以 输出 ，param_name 参数 是 存储 过 程 参 数 名 称 ，type 参数 指定 存储 过 程 
| 的 参数 类 型 ， 该 类 型 可 以 为 MySQL 数据 库 的 任意 数据 类 型 。 
| -个 存储 过 程 包括 名 字 ， 参 数列 表 ， 还 可 以 包括 很 多 SQL 语句 集 。 下 面 创建 一 个 存储 过 
| 程 ， 其 代码 如 下 : 
| delimiter // 

create procedure proc_name (in parameter integer) 

begin 

declare variable varchar(20): 

if parameter=] then 

set variable="MySQL': 

else 

set variable="PHP’; 

endif: 

insert into tb (name) values (variable); 

end; 
| MySQL 中 存储 过 程 的 建立 以 关键 字 create procedure 开始 ， 后 面 仅 跟 存 储 过 程 的 名 称 和 参 
| 数 。MySQL 的 存储 过 程 名 称 不 区 分 大 小 写 ， 如 PROCE10 和 proce10 代 表 同 一 存储 过 程 名 。 存 
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储 过 程 名 或 存储 函数 名 不 能 与 MySQL 数据 库 中 的 内 建 函 数 重 名 。 | 
MySQL 存储 过 程 的 语句 块 以 begin 开始 ， 以 end 结束 。 语句 体 中 可 以 包含 变量 的 声明 、 控 | 
制 语句 、SQL 查询 语句 等 。 由 于 存储 过 程 内 部 语句 要 以 分 号 结束 ， 所 以 在 定义 存储 过 程 前 , 应 | 
将 语句 结束 标志 “:;” 更 改 为 其 他 字符 ， 并 且 应 降低 该 字符 在 存储 过 程 中 出 现 的 机 率 ， 更 改 结束 | 
标志 可 以 用 关键 字 “delimiter” 定 义 ， 例 如 : | 
mysql>delimiter // 
存储 过 程 创建 之 后 ， 可 用 如 下 语句 进行 删除 ， 参 数 proc_name 指 存储 过 程 名 。 


drop procedure proc_name 


2. 应 用 示例 | 
下 面 创建 一 个 名 称 为 count of sutdent 的 存储 过 程 。 首 先 ， 创 建 一 个 名 称 为 students 的 | 
MySQL 数据 库 ， 然 后 创建 一 个 名 为 studentinfo 的 数据 表 。 数 据 表 结构 如 表 4.3 所 示 。 | 


表 4.3 studentinfo 数据 表 结 构 


字 段 名 说 明 
sid | TD | | auo increment | 主键 自 增 弄 sid 
name, VARCHAR(50, | 1 学 生 姓名 
age | vaRCHARGD | | | 学 和 年 具 
sex | VARCHARG) | M | | 学 生性 则 
tel | BoardaD | | | 联系 电话 


创建 一 个 名 称 为 count_of student 的 存储 过 程 ， 统 计 studentinfo 数据 表 中 的 记录 数 。 代 码 | 
如 下 : | 


delimiter // 

create procedure count_of student(OUT count num INT) 
Teads sql data 

begin 

select count(*) into count_num from studentinfo; 

end 

I 


在 上 述 代 码 中 ， 定 义 一 个 输出 变量 count_num。 ES 
存储 过 程 应 用 SELECT 语句 从 studentinfo 表 中 获取 ”网 
记录 总 数 。 最 后 将 结果 传递 给 变量 count num。 存 储 
过 程 的 执行 结果 如 图 4.43 所 示 。 


专家 点 评 图 443 创建 存储 过 程 count of student | 
MySQL 中 使 用 CALL 语句 来 调用 存储 过 程 。 调 用 存储 过 程 后 ， 数 据 库 系统 将 执行 存储 过 | 
程 中 的 语句 。 然 后 将 结果 返回 给 输出 值 。CALL 语句 的 基本 语法 形式 如 下 : | 
CALL sp_name([parameter[,...]]); | 


其 中 sp_name 是 存储 过 程 的 名 称 ，parameter 是 存储 过 程 的 参数 。 
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问题 126 ”如 何 将 Access 数据 导入 SQL Server 数据 库 中 ? 


医 网 问题 前 述 
如 何 将 Access 数据 库 中 的 数据 导入 SQL Server 中 呢 ? 
| 专家 解答 
| 本 例 将 Access 数据 库 文件 db medicine mdb 导入 SQL Server 数据 库 文件 db_database (该 
| 数据 库 为 室 ， 没 有 创建 任何 数据 表 ) 中 ， 步 又 如 下 : 
| (1) 打开 SQL Server 数据 库 ， 右 键 选择 接收 Access 数据 库 文件 的 数据 库 〈 此 处 选择 
| db_database 数据 库 )， 在 弹出 的 菜单 中 选择 “任务 ”一 “导入 数据 ”命令 ， 如 图 4.44 所 示 。 


图 4.44 选择 导入 数据 菜单 


(2) 打开 “SQL Server 导入 和 导出 向 导 ” 窗 口 ， 单 击 “ 下 一 步 ” 按 钮 ， 如 图 4.45 所 示 。 
| (3) 选择 导入 数据 库 的 数据 源 〈 此 处 为 Microsoft Access)， 并 选择 要 导入 的 数据 库 文件 ， 
| 如 图 4.46 所 示 。 


| 
| ”欢迎 使 用 SQL Server 导入 和 导出 向 导 
| 


De 
SS J 


三- 


| 图 445 “SQL Server 导入 和 导出 向 导 ” 窗 口 图 4.46 选择 要 导入 的 数据 库 
了 “16ds 
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(4) 选择 目标 数据 库 文件 ， 如 图 4.47 所 示 。 
(5) 指定 表 复制 或 查询 ， 此 处 选中 “复制 一 个 或 多 个 表 或 视图 的 数据 ” 单 选 按钮 ， 如 | 
图 4.48 所 示 。 | 


扳 定 家 揽 抽 或 间 询 
攻 


图 4.47 选择 目标 数据 库 文件 图 4.48 指定 表 复 制 或 查询 选项 


(6) 选择 要 复制 的 数据 表 〈 此 处 选择 “全 选 )， 单 击 “ 下 一 步 ” 按 钮 ， 如 图 4.49 所 示 。 
(7) 保存 并 立即 执行 ， 单 击 “ 下 一 步 ”按钮 ， 如 图 4.50 所 示 。 


ETTTTTT] 
和 


WPMD 


sr 


ET [wj sm T3573] so | Mm | | 
图 4.49 选择 要 复制 的 数据 表 图 4.50 保存 并 立即 执行 


(8) 验证 向 导 中 选择 的 选项 ， 单 击 “ 完 成 ”按钮 ， 如 图 4.51 所 示 。 
(9) 正在 执行 复制 操作 ， 如 图 4.52 所 示 。 


CEITITT 本 
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图 4.51 验证 向 导 中 选择 的 选项 图 4.52 正在 执行 复制 操作 
"165。 
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(10) 复制 完成 后 ， 单 击 “ 关 闭 ” 按 钮 ， 如 图 4.53 所 示 。 
复制 后 的 db_database 数据 库 中 的 数据 表 文 件 如 图 4.54 所 示 。 


图 4.53 ”数据 表 复制 成 功 图 4.54 复制 后 的 SQL Server 数据 库 文件 
问题 127 如何 解决 PHP 不 支持 SQL Server 的 问题 ? 


| 问题 阐述 

| PHP 有 时 会 与 SQL Server 数据 库 合作 开发 ， 初 次 合作 需要 配置 一 下 ， 和 否则 PHP 会 不 支持 
| SQL Server 服务 器 。 那 么 如 何 解决 PHP 不 支持 SQL Server 的 问题 呢 ? 

| 专家 解答 
(1) 在 php.ini 文件 中 ， 将 “:extension=php_mssql.dll” 前 的 分 号 去 掉 ， 保 存 后 重新 启动 
| Apache 服务 器 。 

| (2) 将 PHP 中 提供 的 ntwdblib.dll 文件 复制 到 本 机 系统 的 windows\system32 文件 夹 下 ， 

| 重新 启动 Apache 服务 器 即 可 。 

| 专家 点 评 

| ntwdblib.dll 文件 是 用 于 与 SQL Server 客户 端的 连接 。 如 果 想 要 PDO 支持 SQL Server 数据 
| 库 , 那么 在 php.ini 文件 中 还 必须 将 “:extension=php_pdo.dll” 和 “:extension=php_pdo_mssql.dll” 
| 前 的 分 号 去 掉 。 


问题 128 ”如 何 连接 带 密码 的 Access 数据 库 ? 


”问题 阐述 
| 虽然 Access 作为 小 型 项 目的 开发 首选 ， 但 是 ， 商 家 不 会 因为 项 目 小 而 不 要 求 它 的 安全 性 。 
"166 


所 以 给 Access 数据 库 设 置 密码 是 十 分 必要 的 。 那 么 , 在 PHP 中 如 何 连 接 带 密码 的 Access 数据 | 
库 呢 ? | 


专家 解答 
1. 设置 Access 数据 库 的 密码 


(1) 设置 Access 数据 库 密码 与 操作 其 他 数据 库 不 同 ， 必 须 以 独占 方式 打开 数据 库 。 
(2) 选择 “工具 ”一 “安全 ”一 “设置 数据 库 密 码 ” 命 令 ， 设 置 密 码 和 确认 密码 ， 如 图 4.55 | 


所 示 。 
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图 4.55 设置 并 确认 密码 


2. 定义 数据 库 的 连接 方法 
<2?php 
$conn = new com("adodb.connection"): 


$connstr="driver={microsoft access driver (*mdb)};Pwd=111; =". realpath("data/db_ | 
databasell.mdb"); | 


$conn->open($connstr); 
?> 


专家 点 评 
当 用 户 对 Access 数据 库 进 行 维护 时 ， 为 了 方便 起 见 ， 可 以 暂时 撤销 Access 数据 库 密 码 ， 


方法 是 : 打开 数据 库 ， 输 入 密码 并 进入 数据 库 ， 然 后 选择 “工具 ”一 “安全 ”一 “撤销 数据 库 
密码 ”命令 ， 根 据 提示 再 次 输入 密码 ， 单 击 “确定 ”按钮 即 可 撤销 数据 库 密码 。 | 


问题 129 MySQL 和 Server 2000 数据 库 中 数据 类 型 的 转换 
问题 阐述 


在 某 些 特殊 的 时 候 , 可 能 要 将 MySQL 数据 库 转换 为 SQL Server 2000 数据 库 , 这 时 间 题 也 
就 随 之 而 来 。 例 如， 在 MySQL 数据 库 中 ,创建 的 字段 使 用 varchar 类 型 ， 该 类 型 的 数据 可 以 进 | 
行 聚合 运算 ， 其 返回 的 值 是 string 类 型 的 数据 。 但 是 ， 在 SQL Server 数据 库 中 ， 创 建 的 字段 如 | 
果 是 varchar 类 型 ， 则 不 能 够 进行 聚合 运算 。 另 外 ， 在 SQL Server 数据 库 中 ，int 型 的 数据 将 不 | 
能 应 用 到 for 循环 语句 中 ， 这 些 都 是 由 于 数据 库 的 转换 而 带 来 的 问题 。 | 
专家 解答 

针对 MySQL 和 Server 2000 数据 库 转 换 中 的 问题 , 其 基本 的 策略 就 是 实现 对 数据 类 型 的 转 
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换 ， 经 过 数据 类 型 的 转换 就 可 以 完成 相应 的 操作 了 。 

例如 ， 在 SQL Server 数据 库 中 创建 的 varchar 类 型 数据 ， 不 能 够 进行 聚合 运算 ， 但 可 以 将 
| 其 转换 为 int 型 进行 计算 。 由 于 int 型 的 数据 不 能 应 用 到 for 循环 语句 中 ， 所 以 必须 将 int 型 的 
会 内 | 数据 转换 为 string 类 型 ， 才 能 应 用 到 for 循环 语句 中 。 
TY | 应 用 示例 
| 在 读 取 SQL Server 数据 库 中 的 数据 时 ,返回 的 访问 记录 结果 集 为 int 类 型 ,因为 在 SQL Server 
| 数据 库 中 int 类 型 的 数据 不 能 应 用 到 for 循环 语句 中 , 所 以 需要 将 该 类 型 的 数据 转换 为 string 类 型 
| 之后， 才能 应 用 到 for 循环 语句 中 ， 然 后 才能 完成 网 站 访问 量 的 统计 输出 ， 其 关键 代码 如 下 : 


<2php 

| /以 图 形 的 形式 输出 数据 库 中 的 记录 数 

| $query="select sum(counts) as ll from tb_count10 "; /查询 数据 库 中 总 的 访问 量 
| Sresults=mssql_query($query,$conn); 

$fwl=mssql_result($results,0,'11"); 


echo "---------- 
/对 补 位 数字 0 的 处 理 
S$len=strlen($fw]); // 获 取 字 符 串 的 长 度 
$str=str_repeat("0",6-$len): // 获 取 6-$len 个 数字 0 
for($a=0;$a<strlen($str);$at+){ /获取 变量 $str 的 字符 串 长 度 


| Sresults=$str[$al: 
| $results='<img src=images/0.gif>"; 


echo $results; // 循 环 输出 $result 的 结果 
} 
/对 数据 库 中 数据 的 处 理 
settype($fwl,"string"); // 进 行 类 型 转换 ,将 INT 类 型 的 数据 转换 为 STRING 类 型 
for($i=0;$i<strlen($fwl);$i++){ ””// 获 取 字符 串 的 长 度 
$result=$fwl[$i]: 
switch($resul) { 


| /如 果 值 为 "0", 则 输出 0.g 计 图 片 
| case "0"; $ret[$1i]="0.gif':break; 
| case "1"; $ret[$1]="1.gif"':break; 
| case "2"; $ret[$i]="2.gif":break:; 
case "3"; $ret[$1]="3.gif"':;break; 
| case "4"; $ret[$1]="4.gif"':break: 
| case "S"; $ret[$i]="5.gif':break; 
case "6"; $ret[$i]="6.gif':break; 
case "7"; $ret[$1]="7.gif"':;break; 
case "8"; $ret[$1]="8.gif"':break: 
case "9"; $ret[$i]="9.gif";break:; 


9 
echo "<img src=images/". $ret[$1].".>"; 
| } 


| ?> 
| 专家 点 评 
实现 数据 类 型 的 转换 可 以 采用 在 变量 前 面 加 上 一 个 小 括号 , 并 把 目标 数据 类 型 写 在 小 括号 
， 中 即 可 。 另 外 ， 也 可 以 通过 settype0 函 数 来 设置 变量 的 类 型 。 其 语法 如 下 : 
| Bool settype ( mixed var, string type ) 

| 。168 。 
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settype() 函 数 的 参数 说 明 如 表 4.4 所 示 。 


表 4.4 settype() 函 数 的 参数 说 明 
参数 说 明 
VaT 必要 参数 。 输 入 的 变量 
必要 参数 。 变 量 的 类 型 。 可 能 值 为 : 
“boolean” (或 为 “bool”， 从 php4.2.0 起 ) 
“integer” (或 为 “int”， 从 php4.2.0 起 ) | 
“float” (只 在 php4.2.0 之 后 可 以 应 用 ， 对 于 旧版 本 中 应 用 的 “double” 现 已 停 用 ) 


type 


“string” | 
“array” | 
“object” | 
“null” (从 php4.2.0 起 ) 
问题 130 PHP 如 何 和 连接 SQL Server 数据 库 ? 
问题 前 述 


PHP 经 常 与 MySQL 数据 库 合 作 ， 但 有 时 也 会 与 SQL Server 数据 库 合作 ， 那 么 在 PHP 中 
如 何 连 接 SQL Server 数据 库 呢 ? | 
专家 解答 | 
PHP 连接 SQL Server 数据 库 的 方法 有 两 种 :一 是 通过 ADO 方式 连接 ， 二 是 通过 mssql | 
connect(O) 函 数 连接 。 下 面 分 别 介绍 这 两 种 连接 方式 。 | 
1. 通过 ADO 方式 连接 SQL Server 数据 库 | 
利用 ADO 方式 建立 与 SQL Server 数据 库 的 连接 ， 与 使 用 ADO 方式 建立 与 Access 数据 库 | 
的 连接 的 方法 类 似 。 建 立 连接 是 在 本 实例 的 conn.php 文件 中 实现 的 ， 代 码 如 下 : | 
<2php 
$conn=new com("adodb.connection' ): 
$connstr="provider=sqloledb:data source=localhost:uid=sa:pwd=;database=db_online":; | 
$conn->open($connstr); | 
?> | 
2. 通过 mssql connect(O 函 数 连 接 SQL Server 数据 库 | 
通过 mssql_connect() 函 数 建立 与 SQL Server 数据 库 的 连接 首先 应 该 配置 php.ini 文件 ,将 | 
extension=php_mssql.dll 前 的 注释 去 掉 ， 然 后 重新 启动 Apache 服务 器 。 通 过 mssql connectO) 函 | 
数 建立 与 数据 库 连接 的 代码 如 下 : | 


<2php | 
S$conn=mssql_connect("localhost","sa",”"): // 建 立 与 SQL Server 数据 库 的 连接 | 
mssql_ select_ db("db_online ",$conn): // 选 择 数据 库 

> 


例如 ， 下 面 使 用 mssql_connect0 函 数 连 接 SQL Server 数据 库 ， 连 接 结果 如 图 4.56 所 示 。 | 
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图 4.56 连接 SQL Server 数据 库 
连接 SQL Server 数据 库 的 代码 如 下 : 


<?php 
$conn=mssql_connect("LLL-SC08Z7OWOV9","sa",""); /建立 与 SQL Server 数据 库 的 连接 
$select=mssql select_db("master",$conn); // 选 择 数 据 库 
if($select){ 
echo" 连 接 成 功 "; 
Jelse{ 
echo" 连 接 失 败 "; 
} 


?> 
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问题 131 
问题 132 
问题 133 
问题 134 
问题 135 
问题 136 
问题 137 
问题 138 
问题 139 
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如 何 获 得 当前 日 期 ? 

如 何 知道 是 星期 几 ? 

如 何 获得 一 个 月 份 的 天 数 ? 
如 何在 MySQL 中 存储 日 期 ? 
如 何 使 用 MySQL 时 间 稚 ? 
如 何 实现 倒计时 ? 

如 何 计算 程序 的 运行 时 间 ? 
如 何 获 取 当 前 系统 时 间 ? 

如 何 实现 网 页 提醒 功能 ? 
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问题 131 如 何 获得 当前 日 期 ? 


问题 阐述 
系统 的 当前 时 间 受 时 区 限制 。 默 认 情 况 下 ， 系 统 的 当前 时 间 是 格林 威 治 时 间 。 那 么 如 何 获 
得 当前 日 期 呢 ? 


专家 解答 


因为 在 PHP 语言 中 ， 日 期 、 时 间 函 数 依赖 于 服务 器 的 地 区 设置 ， 而 PHP 默认 设置 的 是 标 
准 的 格林 威 治 时 间 ( 即 采用 的 是 零 时 区 )， 所 以 ， 如 果 用 户 没有 对 PHP 的 时 区 进行 设置 ， 那 么 
用 户 使 用 日 期 、 时 间 函 数 获取 的 将 是 英国 伦敦 本 地 时 间 〈( 即 零 时 区 的 时 间 )。 这 也 就 是 我 们 为 
什么 要 对 PHP 的 时 区 进行 设置 的 原因 ， 因 为 如 果 不 设置 正确 的 时 区 ， 那 么 PHP 的 日 期 、 时 间 
函数 就 获取 不 到 正确 的 当地 时 间 。 例 如 ， 以 东 八 区 为 例 ， 如 果 当 地 使 用 的 是 北京 时 间 ， 那 么 如 
果 没 有 对 PHP 的 时 区 进行 设置 ， 那 么 获取 的 时 间 就 将 比 当地 的 北京 时 间 少 8 个 小 时 。 
更 改 PHP 语言 中 的 时 区 设置 有 两 种 方法 : 
(1) 在 php.ini 文件 中 ， 定 位 到 [date] 下 的 “:;date.timezone =” 选 项 ， 去 掉 前 面 的 分 号 ， 并 
设置 它 的 值 为 当地 所 在 时 区 使 用 的 时 间 。 修改 内 容 如 图 5.1 所 示 。 
例如 ， 如 果 当 地 所 在 时 区 为 东 八 区， 那么 就 可 以 设置 
“date.timezone = ”的 值 为 : PRC、Asia/Hong Kong、Asia/Shanghai 
(上 海 ) 或 Asia/Urumgqi (乌鲁木齐 ) 等 。 这 些 都 是 东 八 区 的 时 间 。 
设置 完成 后 ， 保 存 文件 ， 重 新 启动 Apache 服务 器 。 
(2) 在 应 用 程序 中 ， 在 日 期 、 时间 函数 之 前 使 用 date_default_timezone_set() 函 数 就 可 以 完 
成 对 时 区 的 设置 。date_default_timezone_set() 函 数 的 语法 如 下 : 
date_default_timezone_set(timezone):; 


参数 timezone 为 PHP 可 识别 的 时 区 名 称 。 如 果 时 区 名 称 PHP 无 法 识别 ， 则 系统 采用 UTC 
时 区 。 

例如 ， 设 置 北京 时 间 可 以 使 用 的 时 区 包括 : PRC (中 华人 民 共 和 国 )、Asia/Chongqing (了 
庆 )，Asia/Shanghai (上 海 ) 或 者 Asia/Urumqi (乌鲁木齐 )， 这 几 个 时 区 名 称 是 等 效 的 。 

获得 服务 器 当前 日 期 的 最 简单 方法 是 使 用 time 函数 。time 返回 当前 日 期 的 UNIX 时 间 戳 。 
可 以 使 用 data 函数 格式 化 日 期 使 其 易于 理解 。 


问题 132 ”如 何 知道 是 星期 几 ? 


图 5.1 设置 PHP 的 时 区 


mh 


问题 讲述 

我 们 按照 星期 来 安排 生活 。 谈 论 日 期 时 ， 经 常 使 用 像 “ 下 个 星期 二 ”或 “上 个 星期 三 ”之 
类 的 短语 。 对 于 我 们 来 说 ， 理 解 这 种 方式 的 日 期 要 比 普通 日 期 容易 很 多 ， 因 为 不 需要 计算 这 个 
日 期 是 不 是 下 个 星期 二 。 因 此 ， 对 于 以 一 种 很 随意 的 格式 给 定 的 日 期 (如 2011 年 6 月 30 日 )， 
如 何 才能 确定 这 个 日 期 是 代表 一 个 星期 中 的 哪 一 天 呢 ? 


es 


第 DD 章 日 期 和 时间 处 理 


专家 解答 


将 日 期 传递 给 strtotime 函数 , 该 函数 具有 理解 日 期 并 将 其 自动 转换 为 UNIX ni 
能 力 ， 这 样 就 可 以 使 用 data 函数 和 1 (小 写 的 L) 占 位 符 转换 这 个 时 间 惟 了 。 
strtotime() 函 数 将 任何 英文 文本 的 日 期 时 间 描 述 解析 为 UNIX 时 间 戳 。 语 法 如 下 : 


int strtotime ( string time [, int now] ) 


strtotime() 函 数 接受 一 个 包含 英语 日 期 格式 的 字符 串 并 尝试 将 其 解析 为 UNIX 时 间 惟 〈 自 | 
January 1 1970 00:00:00 GMT 起 的 秒 数 )， 其 值 相对 于 now 参数 给 出 的 时 间 ， 几时 闪 和 有 民 供 嘴 ， 
参数 则 用 系统 当前 时 间 。 
如 果 参 数 time 的 格式 是 绝对 时 间 ， 则 now 参数 不 起 作用 ; 如 果 参 数 time toi 
间 ， 那 么 其 对 应 的 时 间 由 参数 now 来 提供 。 | 
如 果 解 析 成 功 ， 则 返回 时 间 戳 ， 和 否则 返回 False。 在 PHP 5.1.0 之 前 本 函数 在 失败 时 返回 -1。 | 
下 面 应 用 strtotime 函数 和 date 函数 获取 指定 时 间 的 周 日 期 格式 。 
<2php 
$time=strtotime("Jul 7st 2011"); 
S$weekday=date("l", $time); 


echo $weekday; 
o> 


问题 133 ”如 何 获得 一 个 月 份 的 天 数 ? 


问题 阐述 


获得 一 个 月 份 的 天 数 是 很 常见 的 任务 ， 特 别 是 在 编写 基于 日 期 的 应 用 程序 时 ， 如 日 历 等 。 
那么 如 何 应 用 日 期 函数 获取 一 个 月 份 的 天 数 呢 ? 


专家 解答 
使 用 strtotime 函数 、date 函数 以 及 t 占 位 符 可 以 很 容易 地 获得 这 种 信息 。 其 方法 如 下 : 
<2php 
Stime=strtotime("Jul"); // 返 回 时 间 戳 
S$weekday=date("t", $time):; // 通 过 date 函数 输出 时 间 ， 定 义 格式 为 t 
echo $weekday; 
?> 
问题 134 如 何在 MySQL 中 存储 日 期 ? 
问题 阐述 


可 读 的 日 期 有 很 多 种 格式 ， 能 够 适合 很 多 情况 。 然 而 ， 这 些 格 式 都 不 是 存储 日 期 的 最 好 方 | 

式 。 在 MySQL 中 存储 日 期 的 最 简单 方法 是 按照 其 在 网 页 上 显示 的 方式 存储 ， 例 如 “8 Marth | 

2011”， 采 用 这 种 方式 非常 不 理想 。 | 

因为 ， 在 MySQL 中 运行 的 SQL 语句 的 WHERE 子 句 不 允许 出 现下 面 的 情况 : | 
.173 . 人 


A 人 Bp 六 须知 道 的 300 个 问题 
误 : 
SELECT* FROM table WHERE date >’14 February 2011” 


144 February 2011 不 是 一 个 日 期 值 ， 而 只 是 一 个 字符 串 表示 的 日 期 ， 它 无 法 像 日 期 一 样 操 
作 或 比较 ， 除 非 它 被 转换 为 日 期 值 。 如 果 用 户 以 字符 串 方式 存储 日 期 ， 将 需要 不 停 地 将 它们 在 
字符 串 和 日 期 类 型 的 值 之 间 来 回转 换 ， 这 样 做 是 十 分 让 人 头疼 的 。 那 么 应 该 采用 什么 样 的 格式 
来 存储 日 期 呢 ? 


专家 解答 


最 理想 的 存储 日 期 信息 的 方法 是 使 用 MySQL 时 间 
戳 。 为 了 获得 服务 器 本 地 时 区 的 当前 时 间 ， 可 以 使 用 now 
或 CURRENT TIMESTAMP 函数 ， 也 可 以 使 用 UTC 
TIMESTAMP 函数 获得 UTC 时 区 时 间 戳 。 

通过 MySQL 日 期 时 间 函 数 来 获取 时 间 ， 完 全 符合 
MySQL 数据 库存 储 的 要 求 ， 是 最 为 理想 的 方法 。 例 如 ， 
在 下 面 的 语句 中 分 别 应 用 MySQL 中 的 CURRENT 
TIMESTAMP、now 和 UTC _TIMESTAMP 函数 获取 当前 
日 期 时 间 ， 如 图 5.2 所 示 。 


专家 点 评 图 52 应 用 MySQL 函数 获取 当前 


MySQL 时 间 堆 要 比 UNIX 时 间 鹤 简单 一 些 ， 其 通用 Dae 
形式 是 “YYYYMM-DD HH:MM:SS”, 通常 存储 在 类 型 为 DATETIME (不 要 和 DATE 及 TIME 
弄 混 ， 它 们 分 别 存储 “YYYY-MM-DD” 和 “HH:MM:SS” 形 式 的 值 ) 的 列 中 。 

这 种 形式 的 时 间 堆 适用 于 简单 的 排序 和 比较 操作 ， 而 且 具 有 易 读 性 的 优点 ， 同 时 还 具有 可 
以 预计 的 长 度 ， 这 使 得 它们 比较 容易 验证 。 

可 以 通过 MySQL 的 DATETIME 列 类 型 利用 许多 MySQL 自身 的 日 期 和 时 间 函 数 , 如 果 需 
要 ，DATETIME 列 类 型 也 很 容易 转换 为 UNIX 时 间 稚 。 


问题 133 如何 使 用 MySQL 时 间 疏 ? 


问题 阐述 

尽管 MySQL 时 间 惟 具有 可 读 性 , 但 却 并 不 是 完美 的 , 例如 , 在 网 页 页 面 上 输出 2011-07-09 
16:50:30， 就 显得 过 于 拖 省 ， 相 反 使 用 “February 8 ，2011 at 7 p.m.” 就 显得 工整 时 尚 了 。 那 么 
这 种 格式 应 该 如 何 实现 呢 ? 


专家 解答 


MySQL 像 PHP 一 样 具 有 一 个 日 期 格式 化 函数 DATE FORMAT, 使 用 该 函数 时 ， 只 需要 像 
下 面 这 样 传递 给 它 一 个 格式 字符 串 和 一 个 时 间 戳 即 可 。 


| 
Un 
io 


应 用 DATE FORMAT 函数 转换 时 间 
“174 。 


DATE FORMAT 很 像 PHP 的 date 函数 ， 也 使 用 包含 定义 格式 的 修饰 符 的 格式 字符 串 。 
表 5.1 中 列 出 了 DATE FORMAT 函数 的 常用 修饰 符 。 


表 5.1 DATE_FORMAT 函数 的 常用 修饰 符 


参数 说 明 
%a 星期 的 简写 (Sun~Sat) 
%b 月 份 的 简写 (Jam-~Dec) 
%c 月 ， 数 字 (1~12) 
%d 有 英语 前 缀 的 月 份 的 日 期 
%D 月 份 中 的 天 数 ， 数 字 (00~31) 
%e 月 份 中 的 天 数 ， 数 字 (00~31) 
%I 12 小 时 制 的 十 进 制 小 时 数 〈 范 围 从 00~12) 
%ij 年 份 中 的 第 儿 天 ， 十 进 制 数 (范围 从 001~366) 
%m 十 进 制 月 份 (范围 从 01~12) 
%p 根据 给 定 的 时 间 值 为 am 或 pm， 或 者 当前 区 域 设置 中 的 相应 字符 串 
hr 用 a.m 和 p.m 符号 的 时 间 
$S, 十 进 制 秒 数 
%T 当前 时 间 ， 和 %H:%M:%S 一 样 
%W 本 年 的 第 几 周 数 ， 从 第 一 周 的 第 一 个 星期 一 作为 第 一 天 开始 
9%6W 星期 中 的 第 几 天 ， 星 期 天 为 0 
% 没有 世纪 数 的 十 进 制 年 份 〈 范 围 从 00-99) 
%Y 包括 世纪 数 的 十 进 制 年 份 
9690 文字 上 的 % 字 符 


问题 136 ”如 何 实现 倒计时 ? 


问题 阐述 

在 生活 中 , 我 们 经 常会 看 到 对 期 待 某 一 日 子 或 某 一 时 间 的 到 来 实行 倒计时 计算 。 例 如, 2012 
年 伦敦 奥运 会 、 每 年 春节 晚会 新 的 一 年 到 来 前 等 。 那 么 ， 在 程序 中 如 何 实现 倒计时 呢 ? 
专家 解答 

倒计时 的 原理 非常 简单 , 就 是 用 一 个 固定 的 时 间 减 去 当前 的 时 间 , 所 得 到 的 就 是 剩余 时 间 。 
要 完成 时 间 的 加 减 操作 ， 与 比较 大 小 类 似 ， 都 要 将 时 间 转 换 成 时 间 戳 ， 然 后 才能 计算 ， 最 后 再 
将 时 间 戳 转换 成 日 期 输出 。 

将 时 间 转 换 成 时 间 戳 自然 应 用 的 是 strtotime() 函 数 ， 其 方法 如 下 : 


<?php 
Stimel = strtotime(date( "Y-m-d")): // 当 前 的 系统 时 间 
$time2 = strtotime("2012-1-1"); /2012 年 元 旦 
$sub2 = ceil(($time2 - Stimel) / 86400); /60 秒 *60 分 *24 小 时 ) 秒 /天 


. 175 . 


人 ip 六 预 知道 的 300 个 问题 


Fes 及 
echo "距离 2012 年 元 旦 还 有 <font color=red>$sub2 </font> 天 !1"; 
Te 


会 内 问题 137 ”如 何 计算 程序 的 运行 时 间 ? 


问题 阐述 


| 在 百度 中 ， 当 我 们 执行 一 个 查询 操作 时 ， 获 取 到 查询 结果 后 ， 页 面 中 就 会 出 现 一 行文 字 ， 
| 提示 用 户 根据 关键 字 搜索 到 多 少 个 结果 以 及 搜索 所 用 的 时 间 。 这 个 时 间 就 是 程序 在 执行 这 个 搜 
| 索 时 所 用 的 时 间 ， 那 么 它 是 如 何 实现 的 呢 ? 


| 专家 解答 


| 计算 程序 运行 时 间 的 原理 : 首先， 在 执行 查询 操作 之 前 定义 一 个 时 间 ， 将 其 精确 到 微 秒 ， 
| 然后 ， 执 行 查询 操作 ， 接 着 ， 在 查询 功能 执行 完毕 后 ， 再 次 获取 一 个 时 间 ， 同 样 精确 到 微 秒 ， 最 
| 后 , 应 用 后 获取 的 时 间 减 去 最 初 获取 的 时 间 , 就 是 本 次 查询 所 用 的 时 间 。 其 具体 实现 方法 如 下 : 
| 创建 index.php 文件 ， 首 先 定义 run_time0 方 法 ， 然 后 调用 run_time0 方 法 ， 获 取 当 前 时 间 
| 戳 ， 接 着 运行 程序 ， 最 后 再 次 调用 run_time() 函 数 ， 获 取 系统 当前 时 间 ， 并 且 输 出 差 值 ， 获 取 
| 程序 的 运行 时 间 ， 其 代码 如 下 : 


<2php 

谨 ” ”声明 run_time 函数 

通过 microtimeO 获 取 当 前 时 间 的 微 秒 数 和 时 间 玲 

应 用 explode0 函 数 将 microtime0 函 数 返 回 的 字符 串 进行 分 隔 ， 返 回 一 个 数组 
包括 两 个 元 素 ， 一 个 元 素 是 当前 时 间 的 微 秒 数 ， 另 一 个 元 素 是 当前 时 间 的 时 间 截 
应 用 listO 函 数 将 explode0 函 数 返 回 的 数组 值 赋 给 指定 的 变量 


最 后 获取 两 个 变量 值 的 和 
未 
function run_timeO{ 
list($msec, $sec) = explode(" ", microtime()); // 使 用 explode0 函 数 返回 两 个 变量 
Ietum ((float)$msec + (float)$sec); // 返 回 两 个 变量 的 和 
} 
$start_time = run_time(O; // 第 一 次 运行 ron_time0 函 数 
人 # ”运行 PHP 代 码 段 ”*/ 
$timel = strtotime(date( "Y-m-d")); // 当 前 的 系统 时 间 
$time2 = strtotime("2012-10-1"); /2012 国庆 节 
$sub2 = ceil(($time2 - $timel) /86400): /1(60 秒 *60 分 *24 小 时 ) 秒 /天 
echo "距离 2012 年 国庆 节 还 有 <font color=red>$sub2 </font> 天 !1"; 

素 素 素来 玉 来 玉 束 求 束 束 冰冰 素来 束 求 束 束 */ 
$end time = run_ time(); // 再 次 运行 run_timeO 函 数 
?> 


<!-- 输出 差 值 ， 获 取 程 序 的 运行 时 间 --> 
倒计时 程序 的 运行 时 间 为 <font color=blue> <?php echo ($end_time - $start_time); ?> </font> 秒 
| ?> 
| 专家 点 评 
| 在 计算 程序 的 运行 时 间 过 程 中 ， 应 用 strtotime0 函 数 获取 时 间 戳 ， 实 现 倒计时 功能 ， 应 用 
| microtime() 函 数 将 时 间 精 确 到 微 秒 ， 并 计算 出 程序 的 运行 时 间 。 
了 . 176 . 
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问题 138 ”如 何 获取 当前 系统 时 间 ? 


问题 讲述 

如 何 通过 程序 获取 当前 系统 的 准确 时 间 呢 ? 
专家 解答 

1， 解 决 思路 


获取 系统 当前 时 间 可 以 使 用 date0 函 数 ， 该 函数 用 于 格式 化 一 个 本 地 时 间 / 日 期 ， 语 法 格式 

如 下 : 
string date ( string format , int timestamp) 

该 函数 返回 将 参数 timestamp 按照 指定 格式 字符 串 而 产生 的 字符 串 , 其 中 的 参数 timestamp 
是 可 选 的 ， 默 认 值 为 tme0， 即 如 果 没有 给 出 时 间 戳 则 使 用 本 地 当前 时 间 。 

date() 函 数 的 参数 format 的 格式 化 选项 如 表 5.2 所 示 。 

表 5.2 参数 format 的 格式 化 选项 

参 数 说 明 
小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 
大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 
Swatch Interet 标 准时 ， 返 回 值 为 000-999 
月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 为 01-~31 
星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon~Sun 
月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January~December 
小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1~12 
小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0~23 
小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 01~12 
小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 00~23 
有 前 导 零 的 分 钟 数 ， 返 回 值 为 00~59 
判断 是 否 为 夏令 时 ， 返 回 值 如 果 是 夏令 时 ， 为 1， 否 则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 为 1~31 
星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday~Saturday 
判断 是 否 为 关 年 ， 返 回 值 如 果 是 半年 为 1， 和 否则 为 0 
数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01~12 
3 个 字母 缩写 表示 的 月 份 ， 返 回 值 为 Jan~Dec 
数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1~12 
与 格林 威 治 时 间 相 差 的 小 时 数 ， 如 0200 
RFC 822 格式 的 日 期 ， 如 Thu，21 Dec 2000 16: 01: 07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 为 00~59 
每 月 天 数 后 面 的 英文 后 绥 ， 两 个 字符 ， 如 st、nd、rd 或 了 中。 可 以 和 j 一 起 使 用 


HD Ie lw I> Is 


milo l= lo ls llB IIc |- |- | Is | om 


a 这 癌 


(本 > 六 预 知道 的 300 个 问题 


续 表 


说 明 
指定 月 份 所 应 有 的 天 数 
本 机 所 在 的 时 区 
从 UNIX 纪 元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 
星期 中 的 第 几 天 ， 数 字 表 示 ， 返 回 值 为 0-6 
ISO-8601 格 式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 
4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 
两 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
年 份 中 的 第 几 天 ， 返 回 值 为 0-366 
时 差 偏 移 量 的 秒 数 .UTC 西边 的 时 区 偏 移 量 总 是 负 的 , 而 UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 
返回 值 为 -43200~43200 


SEk<| 有 lsc |- 


说 明 : 

有 效 的 时 间 戳 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 19 日 
03:14:07 (此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 )。 在 Windows 系统 中 此 范围 限制 为 
从 1970 年 1 月 1 日 到 2038 年 1 月 19 日 。 


2. 应 用 示例 
本 示例 主要 应 用 date0 函 数 来 获取 系统 的 当前 时 间 ， 运 2011-06-24 09:07:08 
行 结果 如 图 5.4 所 示 。 
应 用 date0 函 数 获取 系统 的 当前 时 间 ， 时 间 的 格式 为 i 
“yyyy-mm-dd hh:ii:ss”。 其 代码 如 下 : 图 54 获取 系统 当前 时 间 
<2php 
date_default_timezone_set("Asia/Hong Kong"); 
$data=date("Y-m-d h:i:s"); 
echo $data; 
?> 


专家 点 评 


于 在 PHP 语言 中 默认 设置 的 是 标准 的 格林 威 治 时 间 〈 即 采用 的 是 零 时 区 )， 所 以 要 获取 

本 地 当前 的 时 间 必 须 更 改 PHP 语言 中 的 时 区 设置 ， 有 两 种 方法 : (1) 修改 php.ini 文件 中 的 设 

置 ， 找 到 [date] 下 的 “:date.timezone =” 选 项 ， 将 该 项 修改 为 “date .timezone =Asia/Hong Kong”， 

然后 重新 启动 Apache 服务 器 。(2) 在 应 用 程序 中 ， 在 使 用 时 间 / 日 期 函数 之 前 添加 函数 
“date_ default_ timezone_set("Asia/Hong Kong"):” 即 可 。 


有 水 迎 光 临 


问题 139 ”如 何 实现 网 页 提醒 功能 ? 


问题 阐述 
疗 钟 是 生活 中 经 常 使 用 的 一 个 小 工具 ， 它 会 在 指定 时 间 叫 你 起 床 ， 那 么 我 们 是 否 可 以 在 网 
页 中 制作 一 个 闹钟 呢 ? 当 系统 时 间 运 行 到 某 一 时 刻 时 , 为 浏览 者 提供 一 个 温 声 的 提示 信息 呢 ? 
.178 . 
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例如 ， 当 母亲 节 来 临 之 时 ， 访 问 网 站 的 用 户 会 看 到 一 个 温馨 的 提示 ,“ 母 亲 节 到 了 ， 记 得 向 您 
的 母亲 道 一 声 辛苦 了 ”。 | 
专家 解答 


网 页 提醒 功能 ， 通 过 判断 系统 中 当前 的 时 间 与 指定 的 某 个 时 间或 时 间 段 是 否 相 同 ， 如 果 相 | 
同 ， 系 统 就 给 出 一 个 对 应 的 提示 信息 ， 提 示 用 户 该 做 什么 (如 每 天 工作 记录 的 上 传 、 每 月 经 验 | 
技巧 的 提交 或 者 员工 生日 的 提醒 等 )。 | 
上 述 比 较 可 以 有 两 种 方法 ， 第 一 种 方法 是 将 时 间 转 换 为 时 间 戳 ， 然 后 进行 比较 。 例 如 , 当 | 
时 间 运 行 到 9 月 18 日 时 ， 提 醒 千 千 万 万 的 中 华 同胞 们 勿 忘 国耻 。 | 


<2php 
Stimel = strtotime(date("Y-m-d")): /当前 的 系统 时 间 ， 获 取 月 和 天 的 时 间 惟 
Stime2 = strtotime(date("Y™)."-09-18"); /设置 时 间 9 月 18 日 的 时 间 截 
if($timel—$time?2){ // 浏 断 两 个 时 间 惟 是否 相 同 

echo "<script>alert(' 匆 忘 国耻 ! ');window.location.href='index.php';</script>"; /给 出 提示 信息 
} 
9> 


第 二 种 方法 是 应 用 字符 串 函 数 substr0 对 系统 当前 时 间 进 行 截取 , 将 截取 后 的 内 容 与 指定 的 
时 间 进 行 比较 ， 如 果 相同 则 给 出 提示 信息 。 例 如 ， 当 时 间 运 行 到 7 月 1 日 时 ， 提 醒 今 天 是 党 的 | 
生日 。 | 


<?php 
$time = substr(date("Y-m-d"),5,5); /截取 系统 当前 时 间 
if($time—"08-18"){ // 判 断 时 间 是 否 相 同 
echo "<script>alert( 生 日 快乐 ! ");window.location.href='index.php';</script>"，// 给 出 提示 信息 
} 
?> 


gs a 


诬 


图 形 图 像 处 理 


问题 140 
问题 141 
问题 142 
问题 143 
问题 144 
问题 145 
问题 146 
问题 147 
问题 148 
问题 149 
问题 150 
问题 151 
问题 152 
问题 153 
问题 154 


吾 吾 于 于 于 于 于 于 于 于 于 于 于 于 至 


如 何 创建 缩 略图 ? 

如 何 防止 图 像 被 盗 链 ? 

在 PHP 中 如 何 激活 和 屏蔽 GD2 函数 库 ? 

如 何在 照片 上 写 文字 ? 

在 GD2 函数 库 中 ， 如 何 输出 中 文字 符 串 ? 
如 何 应 用 GD2 函数 为 图 片 添加 图 像 水 印 ? 
如 何 应 用 GD2 函数 生成 带 有 干扰 的 图 像 验证 码 ? 
如 何 通 过 图 像 显示 密码 安全 强度 ? 
如 何 从 图 像 中 提取 EXIF 信息 ? 

如 何 应 用 JQuery 和 Jpgraph 制作 动态 折线 图 ? 
如 何 通 过 Jpgraph 类 库 输出 中 文字 符 串 ? 

如 何 制作 数字 图 像 验证 码 ? 

如 何 制 作 中 文 图 像 验证 码 ? 

如 何 实现 验证 码 在 当前 页 验证 ? 

如 何 任意 调整 图 片 的 大 小 ? 
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问题 140 ”如 何 创 建 缩 略 图 ? 


问题 阐述 驮 | 


如 果 某 个 站 点 允许 上 传 图 像 ， 而 这 些 图 像 可 能 与 提交 的 内 容 一 起 显示 ， 该 如 何 保证 显示 的 
图 像 保持 合适 的 大 小 呢 ? 如 果 一 个 用 户 上 传 了 一 幅 特别 大 的 图 像 ， 那 么 它 在 显示 时 可 能 会 对 页 | 
面 布局 造成 破坏 。 | 


专家 解答 


这 个 问题 的 一 种 解决 方案 是 创建 缩 略图 ， 可 以 保证 显示 的 图 像 永 远 不 会 超出 某 个 高 度 和 宽 | 
度 值 。 创 建 一 个 基本 的 缩 略 图 需要 5 个 步 又 : | 
(1) 将 源 图 像 装 载 到 一 个 PHP 变量 中 。 
(2) 确定 原 有 图 像 的 高 度 和 宽度 。 
(3) 创建 一 个 具有 正确 尺寸 的 空白 缩 略图 。 
(4) 复制 原 有 图 像 到 空白 缩 略图 。 
(5) 使 用 正确 的 内 容 类 型 显示 缩 略 图 。 
下 面 为 一 幅 JPEG 格式 的 照片 创建 一 个 缩 略 图 。 
(1) 设置 源 图 像 的 路 径 和 期 望 的 宽度 和 高 度 像素 值 。 
<2php 
$sourceImage = 'images/1.jpg ': 
SthumbWidth = 200; 
S$thumbHeight = 200: | 
(2) 使 用 imagecreatefromjpeg 创建 一 个 jpeg 图 像 ， 通 过 getimagesize 函数 返回 图 像 的 宽 | 
度 和 高 度 。 | 
S$original = imagecreatefromjpeg($sourceImage): 
$dims = getimagesize($sourceImage); | 
(3) 使 用 imagecreatetruecolor 函数 创建 一 个 空白 图 像 (以 PHP 变量 $thumb 的 形式 存在 内 | 
存 中 )， 缩 略图 将 被 放置 在 其 中 。 | 
S$thumb = imagecreatetruecolor($thumbWidth,$thumbHeight); 
(4) 通过 imagecopyresampled 函数 生成 缩 略 图 。 
imagecopyresampled( Sthumib. Soriginal, 0, 0, 0, 0.$thumb Width, Sthumbieight, dims[O], Sdims[1]): | 
imagecopyresampled 函数 将 图 像 的 一 个 调整 大 小 后 的 版 本 放置 到 空白 缩 略图 中 ， 以 这 种 方 | 
法 重新 取样 能 够 确保 图 像 平滑 地 调整 大 小 。 | 
参数 Sthunb 定义 目的 图 像 :参数 $original 定义 源 图 像 。 后 面 的 4 个 参数 代表 目的 图 像 和 源 | 
图 像 部 分 的 和 和 了 立 坐 标 ， 坐 标 系 原点 位 于 左上 角 。 | 
(5) 通过 header0 函 数 定义 内 容 类 型 头 (Content-type: image/jpeg)， 使 用 imagejpegO 函 数 | 
输出 完成 的 缩 略图 : | 
header( "Content-type: image/ipeg" ); 
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a Sthumb ); 
运行 结 效 果 如 图 6.1 所 示 。 
a 


图 6.1 缩 略 图 


问题 141 ”如何 防 止 图 像 被 资 链 ? 


| 问题 阐述 


| 网 站 的 图 片 被 盗 链 是 一 件 让 人 非常 郁闷 的 事情 。 盗 链 不 仅 盗 用 图 片 ， 更 直接 的 问题 是 用 户 
| 在 下 载 盗 链 人 网 站 上 的 图 片 时 ， 会 给 服务 器 带 来 压力 ， 导 致 日 志 中 的 访问 记录 暴涨 ， 而 带宽 被 
| 耗 尽 。 那 么 如 何 才能 防止 图 片 被 盗 链 呢 ? 


专家 解答 


要 防止 网 站 图 片 被 盗 链 ， 那 么 就 有 必要 了 解 盗 链 的 手段 ， 基 本 上 可 以 分 为 两 种 : 

回 ”第 一 种 ， 盗 链 者 直接 使 用 图 片 链接 引用 网 站 上 的 图 片 。 
| 防止 这 种 盗 链 情 况 的 发 生 有 两 种 可 行 的 方法 : 通过 Apache 服务 器 进行 控制 和 使 用 
| SESSION 变量 在 程序 中 进行 控制 。 
| 第 二 种 : 直接 从 网 站 上 下 载 图 片 并 且 复 制 使 用 。 
| 防止 图 像 盗 链 的 方法 主要 有 两 种 解决 方案 , 一 种 使 用 Apache 的 mod_rewrite.so 扩展 , 另 一 
| 种 使 用 PHP 的 SESSION 变量 。 
| 1. Apache 防盗 技术 
| Apache 防盗 技术 的 原理 与 PHP 伪 静 态 技 术 的 原理 是 相同 的 ， 都 需要 应 用 Apache 的 mod_ 
| rewrite.so 模 组 。Apache 服务 器 的 配置 文件 httpd.conf 的 修改 方法 如 下 : 
| (1) 打开 httpd.conf 文件 ， 定 位 到 如 下 位 置 : 

#0oadModule rewrite_module modules/mod _ rewrite.so 


将 该 项 前 面 的 “#” 去 掉 ， 启 动 该 项 。 
| (2) 查找 httpd.conf 文件 ， 找 到 其 中 的 “AllowOverride” 项 ， 将 它 的 值 修改 为 All。 保 存 
| 并 重新 启动 Apache 服务 器 ， 使 修改 生效 。 
| (3) 在 项 目 根 目录 下 创建 .htaccess 文件 ， 定 义 防止 网 站 图 片 被 盗 链 的 方法 。.htaccess 文件 
| 的 代码 如 下 : 
| SetEnvIfNoCase Referer "^http://192.168.1.59/" local ref=1 


<FilesMatch ".(gifjpg)"> 
Order Allow,Deny 
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Allow from env=local ref 
</FilesMatch> 


Referer 字段 : 当 Apache 处 理 一 个 请 求 时 ， 将 检测 头 信息 中 的 Referer 字段 ， 并 且 设 | 
置 环境 变量 local ref 为 1, 如 果 请 求 从 本 身 的 网 站 地 址 开始 , 即 是 本 网 站 的 一 个 页 面 。 | 
^http://192.168.1.59/: 是 一 个 正则 表达 式 , 为 了 设置 环境 变量 , Referer 值 必须 匹配 它 。 | 
"NoCase" 指 令 : 定义 正则 表达 式 的 值 忽略 字符 串 的 大 小 写 。 | 
Order Allow,Deny: 设置 Apache 对 当前 的 请 求 ， 将 执行 列表 中 的 Allow 指令 ,然后 
重复 进行 Deny 指令 。 | 
local ref: 这 将 使 设置 了 local ref 环境 变量 (无论 什么 值 ) 的 请 求 通过 。 任 何其 他 的 | 
请 求 将 被 拒绝 ， 因 为 它们 不 符合 Allow 的 条 件 并 且 默 认 是 拒绝 访问 的 。 | 
2. SESSION 变量 防盗 链 | 
SESSION 变量 防盗 链 技术 的 原理 是 判断 图 片 访问 者 的 权限 , 如 果 有 权限 则 可 以 访问 , 否则 | 
就 不 能 够 访问 。 具体 实 施 就 是 ， 首 先 定义 一 个 SESSION 变量 ， 然 后 使 用 另 一 个 脚本 生成 图 像 ， | 
并 且 在 这 个 脚本 中 判断 这 个 SESSION 变量 是 否 存在 ， 如 果 存 在 则 可 以 访问 图 像 。 例 如 ， 下 面 | 
是 一 个 显示 一 幅 图 像 的 简单 页 面 ， 图 像 的 源 URL 是 一 个 PHP 脚本 ， 确 保 只 有 站 点 的 参观 者 才 | 
能 够 看 到 图 像 。 | 
<2php 
session start(); // 初 始 化 SESSION 变量 
$_SESSION['viewimages']=true; // 定 义 SESSION 变量 ， 并 为 其 赋值 


9> 
<tr> 


加 网 加 


el 


<td colspan="2"> 
<img src="getimage.php?img=bg3_01.jpg" width="267" height="31" alt=""></td> 
<td> 
<img src="getimage.php?img=bg3_02.jpe" width="536" height="31" alt=""></td> 
</tr> 


注意 在 上 面 的 代码 中 登记 了 一 个 称 为 viewimages 的 SESSION 变量 ， 而 <img> 标 记 的 sre | 
属性 中 显示 的 是 getimage.php?img=bg3_01.jpg。 | 
下 面 是 脚本 getimage.php 的 代码 。 首 先 ， 检 查 SESSION 变量 viewimage， 看 其 是 否 被 设置 | 
为 true: | 


<?php 
session start(); // 初 始 化 SESSION 变量 | 
这 (isset($_SESSION[viewimages]) && $_ SESSION[viewimages] 一 tme){ /判断 SESSION 变 | 


量 是 否 为 真 


如 果 SESSION 变量 viewimages 被 设置 为 tue， 那 么 将 会 检查 $_ GET [img'] 变 量 中 提供 的 
图 像 是 否 存在 ， 请 求 有 效 ， 因 此 输出 响应 头 和 图 像 : | 
$dims = getimagesize ( ‘images/ .$_ GET ['img'] ): // 获 取 超 链 接 传递 图 片 的 信息 
header ( 'Content-Disposition: inline; filename= . $_GET [img] ); 
header ( 'Content-Type:' . $dims ['mime'] ): 
header ( 'Content-Length:' . filesize ( "images/ . $_GET [img'] ) ); 
readfile ('images/ . $_GET [img] ); // 完 成 对 图 片 的 读 取 操 作 
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注意 ， 这 段 脚 本 还 使 用 了 getimagesize 函数 获取 图 像 的 尺寸 。 
最 后 ， 通 过 返回 HTTP 状态 “404Not Found” 来 处 理 错 误 情况 : 


else { 
header ( "HTTP/1.1 404 NOT Found" ): 
header ( "Content-Type: text/plain" ); 
echo "图 像 不 存在 ! \n"; 
} 
}else{ 


header ( "HTTP/1.1 404 NOT Found" ); 
header ( "Content-Type: text/plain" ); 
echo "这 是 一 个 被 保护 的 图 像 不 可 盗 链 ! \n"; 
} 
这 种 保护 能 够 阻止 大 多 数 盗 链 者 。 


| 专家 点 评 


使 用 PHP 的 SESSION 变量 防盗 链 ， 这 种 方法 防止 图 片 被 盗 链 操作 起 来 相对 通过 Apache 


| 来 防止 图 片 被 盗 链 要 麻烦 一 些 ， 但 是 如 果 在 购买 的 虚拟 主机 中 使 用 ， 那 么 必须 考虑 虚拟 主机 的 
| Apache 配置 是 否 支持 mod_rewrite.so 模 组 ， 而 使 用 SESSION 变量 来 防止 图 片 被 次 链 ， 则 不 必 
| 考虑 这 个 问题 ， 因 为 它 完全 是 在 用 户 的 程序 中 运行 的 。 所 以 ， 具 体 使 用 哪 种 方法 来 防止 网 站 图 
| 片 被 盗 链 ， 应 根据 用 户 的 实际 情况 而 定 。 


问题 142 ”在 PHP 中 如 何 激活 和 屏蔽 GD2 函数 库 ? 


问题 阐述 


在 PHP 中 如 何 激 活 和 屏蔽 GD2 函数 库 ? 


专家 解答 


GD 库 在 PHP 5 中 是 被 默认 安装 的 ， 但 是 要 想 激 活 GD 库 ， 则 必须 修改 php.ini 文件 。 将 该 


| 文件 中 的 “:;extension=php_gd2.dll” 选 项 前 的 分 号 “;” 去 掉 ， 保存 文件 并 重新 启动 Apache 服务 
| 器 即 可 生效 。 如 果 要 屏蔽 对 GD2 函数 库 的 支持 ， 则 在 “:extension=php_gd2.dll” 选 项 前 加 上 分 
| 号 即 可 。 


问题 143 ”如 何在 照片 上 写 文 字 ? 


| 问题 阐述 


通过 PHP 可 以 输出 照片 ， 那 么 如 何在 照片 上 写 文字 呢 ? 


专家 解答 


在 GD2 函数 库 中 ， 既 可 以 绘制 英文 字符 串 ， 也 可 以 绘制 中 文字 符 串 。 绘 制 英文 字符 串 应 
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用 imagestring0 函 数 。 其 语法 如 下 : 
bool imagestring ( resource image, int font, int x, int y, string s, int col ) 


iagestring0 函 数 用 col 颜色 将 字符 串 s 绘制 到 image 所 代表 的 图 像 的 x，y) 坐标 处 (这 | ， 
是 字符 串 左上 角 坐 标 ， 整 幅 图 像 的 左上 角 为 0，0)。 如 果 font 是 1、2、3、4 或 5， 则 使 用 内 置 | 
字体 。 | 

通过 imagestring0 函 数 水 平地 给 制 一 行 字符 串 “ILike MingRikeJi”。 具体 步 骤 如 下 | 

首先 载 入 一 个 jpg 图 像 ， 然 后 通过 imagestring0 函 数 在 图 片 中 绘制 一 行 英文 字符 叫 。 最 后 | 
答 出 图 像 并 且 销毁 图 像 资源 。 其 代码 如 下 : 


<2php 

header("Content-Type:text/html:charset=utf-8"); // 设 置 页 面 的 编码 风格 
header("Content-Type:image/jpeg"); /告知 浏览 器 输出 的 是 一 张 图 片 
Simg=imagecreatefromipeg("P1020110.jpe"); // 载 入 照片 
$textcolor=imagecolorallocate($img,0,0.0); // 设 置 字 体 颜色 为 蓝 色 ， 值 为 RGB 颜色 值 
imagestring($img,5,80,30,"I Like MingRiKeli", $textcolor):; // 书 写 英文 字符 
imagejpeg($img); // 输 出 图 像 

Imagedestroy($img): /销毁 图 像 

9> 


运行 结果 如 图 6.2 所 示 。 


图 6.2 绘制 英文 字符 串 


问题 144 在 GD2 函数 库 中 ， 如 何 输出 中 文字 符 事 ? 


问题 阐述 
在 GD2 函数 库 中 ， 如 何 输出 中 文字 符 串 ? 
专家 解答 
PHP 中 的 GD 库 支 持 中 文 ， 但 必须 要 以 UTF-8 格式 的 参数 来 进行 传递 ， 如 果 使 用 | 
imageStringO 函 数 直接 绘制 中 文字 符 串 就 会 显示 乱码 ， 这 是 因为 GD2 函数 库 对 中 文 只 能 接收 | 
UTF-8 编码 格式 ， 并 且 默认 使 用 了 英文 的 字体 ， 所 以 要 输出 中 文字 符 串 ， 就 必须 对 中 文字 符 串 | 
进行 转 码 ， 并 设置 中 文字 符 使 用 的 字体 ， 和 否则 ， 输 出 的 只 能 是 乱码 。 | 
PHP 在 图 像 中 添加 中 文字 符 串 应 用 的 是 imagettftext (0) 函数， 其 语法 如 下 : 
array imagettftext ( resource image, float size, float angle, int x, int y, int color, string fontfile, string text ) 
imagettftext 0 函数 的 参数 说 明 如 表 6.1 所 示 。 
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表 6.1 imagettftext () 函 数 的 参数 说 明 
说 明 


图 像 资源 


字体 大 小 。 根 据 GD 版 本 不 同 ， 应 以 像素 大 小 指定 “GD1) 或 点 大 小 (GD2) 
字体 的 角度 ， 顺 时 针 计算 ，0 度 为 水 平 ， 也 就 是 3 点 钟 的 方向 (由 左 到 右 )，90” 则 为 由 下 
到 上 的 文字 


文字 的 x 坐标 值 。 设 定 第 一 个 字符 的 基本 点 


文字 的 ?坐标 值 。 设 定 字 体 基线 的 位 置 ， 不 是 字符 的 最 底 端 


color 文字 的 颜色 
fontfile | 字体 的 文件 名 称 ， 也 可 以 是 远 端 的 文件 
text 字符 串 内 容 


PHP 中 通过 GD2 函数 对 PG 格式 的 照片 进行 操作 ,应 用 imagecreatefromjpeg0) 函 数 ,从 JPEG 
文件 或 URL 新 建 一 图 像 。 其 语法 如 下 : 

Tesource imagecreatefromjpeg ( string filename ) 
参数 filename 可 以 是 本 地 文件 ， 也 可 以 是 网 络 的 URL 地 址 。 返 回 值 为 了 PEG 的 文件 代码 ， 
可 供 其 他 函数 使 用 。 该 函数 在 失败 时 返回 一 个 空 字符 串 ， 并 且 输 出 一 条 错误 信息 。 


应 用 示例 

下 面 通过 GD2 函数 在 图 片 中 输出 中 文字 符 串 。 其 具体 代码 如 下 : 
<2php 
header("content-type:image/jpeg"); // 定 义 输出 为 图 像 类 型 


$im=imagecreatefromjpeg("images/P1020494.JPG"); // 载 入 照片 


Stextcolor=imagecolorallocate($im,25,25,136); 


$font="Font/mzd.ttf"': 
$to=" 沙 起 与 孤 效 齐 飞 "; 


/设置 字体 颜色 为 蓝 色 ， 值 为 RGB 颜色 值 
/定义 字体 


imagettftext($im,80,0,200,200,$textcolor,$font,$to); // 写 TTF 文字 到 图 中 


$to=" 秋 水 共 长 天 一 色 "; 


imagettftext($im,80,0,300,400,$textcolor,$font,$to); // 写 TIF 文字 到 图 中 


imagejpeg($im); // 建 立 JPEG 图 形 
imagedestroy($im); /结束 图 形 ， 释 放 内 存 空间 
?> 


运行 结果 如 图 6.3 所 示 。 


专家 点 评 


划 汪 
gb23 
添加 


由 于 imagettftext() 函 数 只 支持 UTF-8 编码 ， 所 以 在 创 
建文 件 时 必须 也 要 使 用 UTF-8 编码 格式 ， 这 样 才能 保证 中 
符 串 的 正常 输出 。 但 是 ， 如 果 页 面 本 身 使 用 的 是 
12 编码 格式 , 那么 就 必须 


图 63 在 照片 上 输出 中 文字 符 串 
的 中 文字 符 串 进行 编码 格式 的 转换 ， 由 gb2312 编码 转换 为 UTF-8 编码 。 


应 用 iconv0 函 数 对 向 图 片 中 
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问题 145 ”如何 应用 GD2 函数 为 图 片 添加 图 像 水 印 ? 


问题 阐述 

如 何 为 上 传 图 片 添加 一 个 图 像 水 印 ? 
专家 解答 

使 用 图 片 作为 水 印 的 前 提 是 该 图 片 的 背景 必须 是 透明 的 ， 和 否则 输出 的 效果 将 很 不 理想 。 图 
片 水 印 添加 的 关键 是 getimagesize0 和 imagecopy0 函 数 。 应 用 getimagesize() 函 数 获 取 上 传 图 片 | 
和 水 印 图 片 的 大 小 ， 通 过 imagecopy0 函 数 完成 图 片 水 印 的 添加 。 | 

imagecopy0 函 数 将 图 像 复 制 到 指定 的 另外 一 个 图 像 中 。 其 语法 如 下 : 

bool imagecopy ( resource dst_im, resource SIc_im int dst_x, int dst_y., int sre_x, int sre_y, int sre_w, int sre h) 


将 sre_im 图 像 中 坐标 从 sre_ x，src_y 开始 ， 宽 度 为 src_w， 高 度 为 src_ h 的 一 部 分 复制 到 
dst_im 图 像 中 坐标 为 dst_x 和 dst_y 的 位 置 上 。 | 
应 用 示例 | 
在 本 示例 中 ， 整 合 图 像 的 创建 、 操 作 、 输 出 和 销毁 函数 ， 创 建 AddWaterPress 类 ， 定 义 | 
getExtendsName() 方 法 ， 获 取 上 传 图 片 的 文件 后 缀 ;定义 getImageRes0 方 法 ， 根 据 上 传 文件 的 
后 级 创建 新 图 像 ， 定 义 outputImage() 方 法 ， 输 出 图 像 ， 定义 add0 方 法 ， 向 指定 的 图 片 中 添加 | 
图 像 水 印 。 其 具体 实现 过 程 如 下 : | 
(1) 创建 index.php 文件 。 首 先 ， 创 建 一 个 表单 ， 完 成 上 传 图 片 的 提交 操作 。 然 后 ， 在 本 | 
页 中 输出 存储 在 服务 器 指定 文件 夹 下 的 图 片 。 最 后 ， 获 取 表单 中 提交 的 图 片 数 据 ， 应 用 | 
move_uploaded_file0 函 数 完成 图 片 的 上 传 操作 ， 并 且 实 例 化 存储 在 AddWaterPress.php 文件 中 | 
的 AddWaterPress 类 ， 调 用 add0 方 法 为 指定 的 图 片 添加 文字 水 印 。 其 关键 代码 如 下 : | 
if(move_uploaded file($_FILES["file"]["tmp_name"], $saveDir)){ ”// 执 行文 件 上 传 操作 


require_once 'AddWaterPress.php': // 包 含 添加 水 印 操作 的 文件 
$addWaterPress = new AddWaterPress():; // 类 的 实例 化 
$addWaterPress->add($saveDir, "吉林 省 明日 科技 "): // 执 行 添加 方法 ， 传 递 参数 ， 
指定 水 印 文字 


echo "<script>alert( 图 片 添加 成 功 "):</script>"; 
} | 
(2) 在 编写 AddWaterPress 类 的 add() 方 法 时 ， 应 用 的 是 getimagesize() 和 imagecopy0 函 数 
完成 图 像 水 印 的 添加 操作 。Add0 方 法 的 代码 如 下 : 


function add($imageUrl, $watherImageUrl, $x, $y){ // 定 义 添加 方法 
Simg = @$this->getImageRes($this->getExtendsName($imageUrl), $imageUrl): 


// 获 取 被 添加 的 图 像 标 识 
$imgl = @$this->getImageRes($this >getExtendsName($watherImageUil), $watherImageUl): 
S$size = getimagesize($imageUrD): // 获 取 图 像 大 小 
$sizel = getimagesize($watherImageUrD: /获取 水 印 图 片 的 大 小 
if($x—null && $y 一 aulD){ // 浏 断 参数 是 否 为 空 


S$xl = ($size[0]-$sizel[0])/2: /根据 图 像 大 小 数组 中 返回 的 值 ， 计 算 图 像 的 横 坐标 | 
$y1 = ($size[1]-$sizel[1D/2; /根据 图 像 大 小 数组 中 返回 的 值 ， 计 算 图 像 的 纵 坐 标 | 
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jelse{ 


$1 = $x // 如 果 不 为 空 ， 则 直接 使 用 坐标 数据 
$yl = $y; // 如 果 不 为 空 ， 则 直接 使 用 坐标 数据 
| } 
| imagecopy(Simg, Simg1, $x1, $y1, 0, 0, $size1[0], $sizel[1]); /将 imgl 的 一 部 分 拷贝 到 img 
缚 -| 的 指定 位 置 
| /根据 图 像 标识 符 、 后 绥 和 路 径 ， 执 行 outputImage 方法 ， 输 出 图 像 
S$this->outputImage(Simg, Sthis->getExtendsName(SimageUrD, 9imageUrD): 
| imagedestroy($img1); // 销 毁 图 像 
imagedestroy($img): // 销 毁 图 像 
其 运行 结果 如 图 6.4 所 示 。 


| 专家 点 评 


| 在 将 图 片 作为 水 印 图 像 时 ， 如 果 指 定 的 图 片 背景 不 是 透 
| 明 的 ， 那么 添加 的 水 印 图 像 将 与 原始 图 像 格格 不 入 ， 而 如 果 
| 水 印 图 片 的 背景 是 透明 的 ， 那 么 添加 的 水 印 标记 就 可 以 很 好 
| 地 与 原始 图 像 融 入 一 体 。 


图 6.4 为 图 片 添加 图 像 水 印 


| 问题 146 ”如 何 应 用 GD2 函数 生成 带 有 和 干扰 的 图 像 验证 码 ? 


| 问题 阐述 

| 在 用 户 注册 时 ， 添 加 完 基本 注册 信息 后 ,通常 会 被 要 求 输入 验证 码 ， 而 为 了 有 效 增加 恶意 
| 程序 探测 验证 码 的 困难 程度 ， 常 常会 在 验证 码 中 添加 一 些 干扰 元 素 ， 如 干扰 点 、 干 扰 线 等 ， 这 
| 些 干扰 点 或 干扰 线 是 如 何 制 作 的 呢 ? 


| 专家 解答 
| 绘制 干扰 点 主要 应 用 到 imagesetpixel0 函 数 ， 该 函数 的 语法 如 下 ; 
bool imagesetpixel (resource image, int x, int y, int color ) 


参数 说 明 : 

@ image: 图 像 标识 。 

@x: 点 的 xX 坐标 。 

© Ve 点 的 ¥ 坐标 。 

@ color: 指定 点 的 颜色 。 

例如 ， 用 imagesetpixel0 函 数 在 图 像 上 绘制 5 个 相连 接 的 点 ， 代 码 如 下 : 
Sim=imagecreatefromipeg("bg.jpe"): 
$color=imagecolorallocate($im,255,255,255):; 
imagesetpixel ($im,100,80,$color); 
imagesetpixel ($im,100,81,$color); 
imagesetpixel ($im,100,82,$colon); 
imagesetpixel ($im,100,83,$color); 
imagesetpixel ($im,100,84.,$color); 

| imagejpeg($im); 
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本 示例 的 运行 结果 如 图 6.5 所 示 。 

应 用 示例 

本 示例 应 用 GD2 函数 绘制 图 像 验证 码 , 并 通过 JavaScript 调用 所 绘制 的 验证 码 显 示 在 用 户 
注册 表单 中 ， 结 果 如 图 6.6 所 示 。 


~ 


连续 绘制 5 个 点 


同 广 征 第 有 rrrS 扣 结 务 过 各 ， 必 太志 洲 | 
Bi 


4 入 全 问 二 法 目的 而 全 及 6 拉 各 生 ， 


ET HY, #7 所 不 短 全 全 可 人 的 天 20， 中 仙 他 人 
的 ， 守 要 性 的 ， 允 古寺 的 ， 记 伦 潭 种 或 瑟 世 任 问 丰 
[jms en 届 


上 和 条 名 克 定 和 ee4 利 撤 限 公司 
公司 地 站 丙丁 和 长春 币 二 而 Ewei99 吐 


图 6.5 ”imagesetpixel0 函 数 在 图 像 上 连续 绘制 5 个 点 图 6.6 带 有 干扰 线 的 验证 码 

通过 JavaScript 脚本 生成 的 4 位 随机 数 作为 验证 码 的 值 ， 并 且 通 过 JavaScript 脚本 中 的 方 
法 完成 验证 码 的 二 次 生成 。 当 用 户 单 击 “ 看 不 清 ” 超 链接 时 ， 不 刷新 页 面 而 重新 生成 验证 码 ， 
并 将 验证 码 的 值 赋 给 隐藏 域 。 其 关键 代码 如 下 : 


<script language="javascript"> 
var num1=Math.round(Math.random()*10000000); 


var num=numl.toStringO.substr(0.4): /产生 4 位 随机 数 

document.write("<img name=codeimg src="ValidatorCode.php?code="+num+">"); 

forml.defValidatorCode.value=num: // 将 随机 数 赋 给 隐藏 域 
function reCodeO{ // 该 方法 用 于 重新 生成 验证 码 


var num1=Math.round(Math.random()*10000000); 
var num=numl.toStringO.substr(0.4); 
document.codeimg.srce="ValidatorCode.php?code="+num 
form!] .defValidatorCode.value=num: 
} 
</script> 
通过 GD2 函数 将 JavaScript 脚本 生成 的 4 位 随机 数 绘 制 成 图 像 , 并 且 为 验证 码 的 图 像 添 加 
干扰 点 。 其 中 添加 干扰 点 的 代码 如 下 : 
for ($i= 0; $i < 200; $i++) { // 绘 制 200 个 干扰 点 
imagesetpixel($im, rand() % 70, rand() % 30, imagecolorallocate($im, rand(0, 255), rand(0, 255), 
rand(0, 255))): 
) 


问题 147 ”如何 通过 图 像 显 示 密 码 安 全 强度 ? 


问题 讲述 
在 注册 页 设置 密码 时 ， 常 常会 有 图 像 显示 所 设 密码 的 强度 ， 这 是 如 何 实现 的 呢 ? 
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专家 解答 


对 密码 的 安全 强度 进行 判断 是 在 JavaScript 脚本 中 应 用 正则 表达 式 来 完成 的 。 其 原理 如 下 : 
(1) 对 密码 中 的 值 进行 判断 ， 如 果 只 包含 数字 、 英 文字 符 串 或 特殊 字符 串 中 的 任意 一 种 ， 


| 那么 说 明 密码 安全 强度 差 。 


(2) 对 密码 中 的 值 进行 判断 ， 如 果 包 含 数字 、 英 文字 符 串 或 特殊 字符 串 中 的 任意 两 种 ， 
那么 说 明 密 码 安全 强度 中 。 

(3) 对 密码 中 的 值 进行 判断 ， 如 果 同 时 包含 数字 、 英 文字 符 串 和 特殊 字符 串 ， 那 么 说 明 
密码 安全 强度 强 。 

在 JavaScript 脚本 中 , 通过 search0 方 法 调用 正则 表达 式 完成 对 密码 值 的 判断 ， 并且 根 据 不 
同 的 值 定义 输出 不 同 的 图 像 ， 其 关键 代码 如 下 : 


var pl = (form.pwd.value.search(/[a-zA-2Z]/)!=-1)?1:0; /判断 密码 中 是 否 包含 英文 字符 串 
var p2 = (form.pwd.value.search(/[0-9]/)!=-1)?1:0; // 判 断 密码 中 是 否 包 含 数 字 
Var p3 = (form.pwd.value.search(/[^a-zA-Z0-9_]/)!=-1)?1:0; /判断 密码 中 是 否 包 含 特殊 字符 
var p= pl+p2+p3; /定义 判断 的 返回 值 
这 p 一 1){ 
chknew_pwd.inner] ="<img src='images/1_03.jpg>"; // 输 出 密码 安全 强度 差 的 图 像 
ele te 
chknew_pwd.innerHTMIL="<img src='images/2_03.jpg>": // 输 出 密码 安全 强度 中 的 图 像 
Mate 
chknew_pwd.inner] 二 "<img src='images/3_03.jpg>"; // 输 出 密码 安全 强度 强 的 图 像 
} 
应 用 示例 


在 本 示例 中 实现 一 个 用 户 注 册 功 能 ， 通 过 JavaScript 脚本 判断 密码 值 的 强度 ， 并 且 以 图 像 
的 形式 进行 输出 。 其 具体 步骤 如 下 : 

(1) 创建 index.php 页 面 ， 完 成 用 户 注册 页 面 和 表单 的 设计 ， 通 过 JavaScript 对 表单 中 提 
交 的 值 进行 判断 ， 并 直接 将 判断 的 结果 反馈 到 div 标签 中 ， 最 终 将 表单 中 的 数据 提交 到 index_ 
ok.php 文件 中 ， 完 成 用 户 注册 的 操作 。 其 关键 代码 如 下 


<form name="form reg” method="post" action="index ok.php" onSubmit="return chkreginfo 
(form reg,'all)"> 
<tr> 
<td height="30"><div align="right"> 密 码 : </div></td> 
<td height="30" co ="2" align="left">&nbsp: 
<input type="password" name="pwd" id="pwd" size="30" class="inputcss" 
onBlur="chkreginfo(form reg,1)"> <font color="#FF0000">*</font>&nbsp; 
<div id="chknew_pwd" style="color: #FF0000"></div> 
</td> 
</t> 
<tr> 
<td height="30">&nbsp;</td> 
<td width="150" height="30"><input type="image" src="images/form (2).jpg"> &nbsp; 
&nbsp:</td> 
<td width="343"><img src="images/form.jpg" width="72" height="26" 
‘onClick="form reg.reset|" style="cursor: hand" /></td> 
</tr> 
</form> 
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(2) 编写 JavaScript 脚本 ,创建 checkjs 脚本 文件 ， 完 成 对 表单 中 提交 的 值 的 判断 。 

(3) 创建 index_ok.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 将 数据 添加 到 指定 的 数据 表 中 。 | 

运行 本 示例 ， 当 在 密码 文本 框 中 输入 一 个 密码 时 ， 光 标 离 开 后 ， 将 自动 判断 这 个 密码 的 安 | 
全 强度 ， 并 以 图 像 的 形式 返回 一 个 判断 结果 。 其 运行 结果 如 图 6.7 所 示 。 | 


i 
Se 后 了 明日 编程 


首页。 明日 图 书 。 编 科 词 册 ”明日 在 线 支付 


口 月 户 注册 


:3134972256 


上 
志 活 码 0 检 式 不 正确 


Ez 
图 6.7 通过 图 像 显示 密码 安全 强度 
专家 点 评 


在 本 实例 中 ， 不 止 一 次 使 用 了 正则 表达 式 ， 分 别 通过 正则 表达 式 对 密码 的 强度 进行 判断 ， 
对 手机 号 码 进行 判断 ， 对 座机 号 码 进行 判断 以 及 对 邮箱 地 址 进行 判断 。 


/^13Qd{9})$|/^15Qd{9})$|^1890d{8})$/ // 判 断 手机 号 码 
/人 ^Qd{3}-)Qq{8}))SFQd{}-)Qd{7))SIMQqd{4}-)Qq{8))$/ /1/ 判 断 座 机 号 码 
Nw+([-+." wt)*@Ww+([-. J wt) wt([-. ws // 判 断 邮箱 


问题 148 ”如 何 从 图 像 中 提取 EXIF 信息 ? 


问题 阐述 | 

EXIF 信息 就 是 由 数码 相机 在 拍摄 过 程 中 采集 的 一 系列 信息 ,然后 把 信息 放置 在 JPEG/TIFF 
文件 的 头 部 ， 也 就 是 说 ，EXIF 信息 是 镶嵌 在 JPEG/TIFF 图 像 文件 格式 内 的 一 组 拍摄 参数 , 主 | 
要 包括 摄影 时 的 光圈 、 快 门 、ISO、 日 期 时 间 等 各 种 与 当时 摄影 条 件 相 关 的 信息 ， 相 机 品牌 型 | 
号 ， 色 彩 编码 ， 拍 摄 时 录制 的 声音 以 及 全 球 定位 系统 (GPS) 等 信息 。 既 然 在 EXIF 中 包含 了 | 
如 此 详细 和 完整 的 信息 ， 那 么 在 PHP 中 应 该 如 何 获取 到 这 些 信息 呢 ? | 
专家 解答 | 

在 PHP 中 获取 这 些 信息 应 用 的 是 EXIF 函数 。 为 了 使 用 EXIF 函数 ， 必 须 保 证 PHP 已 经 启 | 
用 了 EXIF 支持 。 在 php.ini 文件 中 查看 “extension=php_exifdll” 前 是 否 有 分 号 “:” 如 果 有 则 | 
说 明 未 加 载 ， 那 么 就 要 去 掉 分 号 。 | 

同时 还 要 注意 “extension-php_exif.dll” 项 的 存储 位 置 ， 它 必须 放置 在 “extension-php_ | 
mbstring.dll” 项 之 后 ， 奋 则 EXIF 函数 是 不 被 支持 的 。 设 置 完成 后 保存 php.ini 文件 ， 并 重新 启 ， 
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可 
动 Apache 服务 器 。 然 后 就 可 以 应 用 EXIF 函数 库 中 的 函数 获取 图 像 的 EXIF 信息 。 

获取 图 像 的 EXIF 信息 应 用 的 是 EXIF 函数 库 中 的 exif read_data0 函 数 。 其 语法 如 下 : 
| alTay exif read data ( string filename [, string sections [, bool arrays [, bool thumbnail]]] ) 
| 参数 说 明 ; 
| @ filename: 被 读 取 的 图 像 文件 名 。 不 可 以 是 URL。 
”2% | @ sections: 定义 存在 于 文件 中 的 逗号 分 隔 的 区 段 列 表 用 来 产生 结果 数组 。 如 果 未 找到 所 


| 请 求 的 区 段 则 返回 值 为 False。 该 参数 所 包含 的 区 段 如 表 6.2 所 示 。 
表 6.2 exif_read_data() 函 数 中 参数 sections 包含 的 区 段 


区 段 名 称 包含 内 容 、 说 明 
FILE FileName、FileSize、FileDateTime、SectionsFound 


html、Width、Height、IsColor， 可 能 有 更 多 其 他 的 。Height 和 Width 是 用 和 getimagesize() 
COMPUTED | 一 样 的 方法 计算 的 ， 因 此 它们 的 值 不 能 是 任何 返回 的 头 信息 的 部 分 。 此 外 ，html 是 一 个 
height/width 的 文本 字符 串 ， 可 以 用 于 普通 的 HTIML 中 

ANY TAG | 任何 包含 有 标记 的 信息 ， 如 IFD0、EXIF.… 

IFD0 所 有 IFD0 的 标记 数据 。 在 标准 的 图 像 文 件 中 包含 了 图 像 大 小 及 其 他 

THUMBNAIL| 如 果 有 第 二 个 IFD, 则 文件 应 包含 有 缩 略图 。 所 有 有 关 霸 入 缩 略 图 的 标记 信息 都 存储 在 本 区 
COMMENT |JPEG 图 像 的 注释 头 信息 

EXIF 区 段 是 DO 的 子 区 ， 包 含有 图 像 的 更 多 详细 信息 。 大 多 数 内 容 都 是 与 数码 相机 


Ep 相关 的 


@ arrays: 指定 是 否 每 个 区 段 都 成 为 一 个 数组 。sections COMPUTED、THUMBNAIL 和 
COMMENT 区 段 总 是 成 为 数组 ， 因 为 它们 里 面包 含 的 名 字 和 其 他 区 段 冲突 。 

@ thumbnail: 如 果 该 值 为 True， 读 取 缩 略图 本 身 ， 否则 只 读 取 标记 数据 。 

该 函数 的 返回 值 是 一 个 关联 数组 ， 键 名 是 头 信息 名 ， 值 是 与 其 对 应 的 数据 。 如 果 没 有 可 供 
返回 的 数据 ， 则 exif read_data() 将 返回 False。 

应 用 示例 
| 下 面 应 用 exif read_data() 函 数 获取 图 像 的 EXIF 信息 。 在 分 页 输出 上 传 图 片 时 ， 运 行 自 定 
| 义 函 数 GetImageInfo 获取 图 片 的 EXIF 信息 ， 并 且 输 出 图 片 的 名 称 和 拍摄 时 间 。 
| (1) 在 index.php 页 面 中 ， 实 现 上 传 图 片 的 分 页 输出 ， 并 且 包 含 exifphp 文件 ， 调 用 自 定 
| 义 函数 GetImageInfo 获取 图 片 的 EXIF 信息 ， 输 出 图 片 的 名 称 和 拍摄 时 间 。 其 关键 代码 如 下 : 


| <2php 
| include "exif php"; // 包 含 exifphp 文件 
| S$pic=GetImageInfo($myrow[file_test]): // 调 用 getimageinfo 方法 , 获取 原始 图 片 的 exif 信息 
| echo "文件 名 : ".$pic[ 文 件 名 ']."<br>"; // 输 出 图 片 名 称 
echo "拍摄 时 间 :".$pic[' 拍 摄 时 间 ']: // 输 出 拍摄 时 间 


?> 
(2) 创建 exifphp 文件 。 编 写 自 定义 函数 getimageinfo()， 应 用 exif read_data() 函 数 获取 
| 指定 图 片 的 ex 站 信息 。 其 关键 代码 如 下 : 
| <?php 
| function GetImageInfo($img) { 
j 。192 。 
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Simgtype = array (", "GIF", "JPG", "PNG", "SWF", "PSD", "BMP" ); 
S$Orientation = array ("", "top left side", "top right side", "bottom right side" ); 
SResolutionUnit = array ("" ", "英寸 " "厘米 " ); 
Sexif = exif read data ( $img, "IFDO" ); 
if ($exif —= false) { 
Snew_img info =array ("文件 信息 " => "没有 图 片 EXIF 信息 " ); 
}else{ 
Sexif = exif read data ( $img, 0, tme ); 
Snew_img info = array ("文件 名 "=> Sexif [FILE] [FileName], "时 间 蕉 " => date | 
("Y-m-d H:i:s", $exif [FILE] [FileDateTime] ): | 
四 


Ietum $new_img info; 


} 
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问题 阐述 
如 何 应 用 JQuery 和 Jpgraph 制作 动态 折线 图 ? 
专家 解答 | 
应 用 JQuery 和 Jpgraph 制作 动态 折线 图 主要 是 应 用 JQuery 中 的 属性 实现 数据 的 动态 获取 ， | 
应 用 Jpgraph 生成 折线 图 。 | 
1.JQuery 中 的 属性 函数 
attr(key,fn) 为 所 有 匹配 的 元 素 设 置 一 个 计算 的 属性 值 。 其 不 提供 值 ， 而 是 提供 一 个 函数 ， 
由 这 个 函数 计算 的 值 作为 属性 值 。 | 
参数 key(String) 表 示 属 性 名 称 ， 参 数 甸 (Function) 表 示 返 回 值 的 函数 。 其 范围 是 当前 元 素 ，| 
参数 是 当前 元 素 的 索引 值 。 | 
示例 : 把 src 属性 的 值 设置 为 title 属性 的 值 。 


<img STc="testjpg"/> 
S$("img").attr("title", fonction0 { return this.src }); 


输出 结果 : 
<img strc="test.jpg" title="testjpg" /> 
attr(keyvalue)， 为 所 有 匹配 的 元 素 设置 一 个 属性 值 。 
参数 key (String) 是 属性 名 称 ; 参数 value (Objecb 是 属性 值 。 
示例 : 为 指定 图 像 设 置 src 属性 。 


<img id="img" sre="img php?m=0" 及 
function beginProgress(counts){ // 创 建 自 定义 函数 
$("#img").attr("sre", "img.php?m="+counts):; /为 图 像 设 置 src 属性 


attrtname)， 取 得 第 一 个 匹配 元 素 的 属性 值 。 通 过 这 个 方法 可 以 方便 地 从 第 一 个 匹配 元 素 中 
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获取 一 个 属性 的 值 。 如 果 元 素 没有 相应 属性 ， 则 返回 undefined。 
参数 name (String) 是 属性 名 称 。 
示例 : 返回 页 面 中 第 一 个 图 像 的 src 属性 值 。 
<img sTc="testjpg"/> 
$("img").attr("sre"); 
输出 结果 : testjpg 
attr(properties)， 将 一 个 “名 / 值 ” 形式 的 对 象 设置 为 所 有 匹配 元 素 的 属性 。 这 是 一 种 在 所 有 


| 匹配 元 素 中 批量 设置 很 多 属性 的 最 佳 方式 。 


| 说明 : 


参数 properties (Map) 是 作为 属性 的 “名 / 值 对 ”对 象 。 
示例 : 为 所 有 图 像 设置 sc 和 alt 属性 。 
$("img").attr({ src: "test.jpg", alt: "Test Image" }); 


如 果 要 设置 对 象 的 class 属性 ， 则 必须 使 用 className 作为 属性 名 ， 或 者 可 以 直接 使 


| 用 .addClass( class ) 和 .removeClass( class )。 


removeAttr(name)， 从 每 一 个 匹配 的 元 素 中 删除 一 个 属性 。 
参数 name (String) 是 要 删除 的 属性 名 。 
示例 : 将 页 面 中 图 像 的 src 属性 删除 。 


<img STc="testjpg"/> 
$("img").removeAttr("sre"); 


2，Jpgraph 下 载 与 安装 
要 想 应 用 Jpgraph， 首先 要 下 载 并 安装 Jpgraph， 同 时 下 载 该 类 库 的 压缩 包 。Jpgraph 类 库 的 


压缩 包 主 要 有 两 种 形式 ， 即 zip 格式 和 tar 格式 ， 如 果 是 Linux/UNIX 平台 可 以 选择 tar 格式 的 
| 压缩 包 ; 如 果 是 微软 的 Win32 平台 选择 上 述 两 种 格式 的 压缩 包 任 一 种 都 可 以 。Jpgraph 类 库 可 


以 从 其 官方 网 站 “http://www.aditus.nu/jpgraph/” 下 载 ， 目 前 最 新 的 版 本 是 2.3。 

Jpgraph 需要 GD 库 的 支持 。 如 果 用 户 希 望 Ipgraph 类 库 仅 对 当前 站 点 有 效 , 只 需 将 Jpgraph 
压缩 包 下 的 src 文件 夹 中 的 全 部 文件 复制 到 网 站 所 在 目录 下 即 可 ， 在 使 用 时 ， 只 需 调用 src 文 
件 夹 下 的 指定 文件 即 可 。 


说 明 : 

如 果 想 要 让 整个 PHP 开发 环境 支持 Jpgraph， 那 么 可 以 使 用 下 面 的 配置 方法 。( 1 ) 将 压缩 
包 下 的 全 部 文件 解压 到 一 个 文件 夹 中 ， 如 “F:\AppServiwww\jpgraph”。(2) 打开 PHP 的 安装 目 
录 ， 编 辑 php.ini 文件 并 修改 其 中 的 include path 参数 ， 在 其 后 增加 前 面 的 文件 夹 名 ， 如 
include path = ".:F:\AppServ\wwwAjpgraph"。(3 ) 重新 启动 Apache 服务 器 即 可 生效 


3. 应 用 示例 
本 示例 应 用 jQuery 和 人 pgraph， 开 发 一 个 能 够 随时 更 新 的 动态 折线 图 ， 通 过 该 折线 图 分 析 
网 站 当月 访问 量 的 走势 。 其 运行 结果 如 图 6.8 所 示 。 
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图 6.8 ”应 用 JQuery 和 Jpgraph 制作 动态 折线 图 


(1) 创建 index.php 文件 ， 以 数字 图 片 的 形式 输出 网 站 总 访问 量 ， 并 创建 “网 站 访问 量 分 
析 ” 的 超 链 接 。 其 关键 代码 如 下 : | 


td height="50" align="center"> 
<2php 
/以 图 形 的 形式 输出 数据 库 中 的 记录 数 
$query="select sum(counts) as 11 from tb_count10"; /查询 数据 库 中 总 的 访问 量 
Sresult=mysql_query($query); 
S$fwl=mysql_result($result,0,"11"): 
ee 机 


Slen=strlen($fwl); // 获 取 字 符 串 的 长 度 
$str=str_repeat("0",6-$len); /获取 6-$len 个 数字 0 
for($i=0:$i<strlen($str):$i++){ // 获 取 变量 $str 的 字符 串 长 度 

$result=$str[$i]; 

Sresult="<img src=images/0.gif> 

echo $result: 1/ 循环 输出 $result 的 结果 


} 

// 对 数据 库 中 数据 的 处 理 

for($i=0;Si<strlen($fwl):$i++){ // 获 取 字 符 串 的 长 度 
$result=$fwl[$1]; 

switch($result){ 

// 如 果 值 为 "0", 则 输出 0.g 下 图 片 
case "0"; $ret[$1]="0.gif"':break; 
case "1"; $ret[$1]="].gif":;break; 
case "2"; $ret[$i1]="2.gif':break:; 
case "3"; $ret[$1]="3.gif':break: 
case "4"; $ret[$1]="4.gif':break:; 
case "5": $ret[$i]="5.gif":break: 
case "6"; $ret[$1]="6.gif':break: 
case "7": $ret[$i]="7.gif":break: 
case "8": Sret[$i]="8.gif':break: 
case "9": Sret[$i]="9.gif":break: 
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| echo "<img src=images/". $ret[$1].".>"; // 输 出 访问 次 数 
} 


| 2> 
<a href="search.php" class="STYLE1"> 网 站 访问 量 统计 分 析 </a> 


4 | </td> 
天 A 


(2) 创建 search.php 文件 。 对 网 站 总 的 访问 量 、 总 的 他 访问 量 ， 当 月 访问 量 、 当 月 名 访 
问 量 ， 当 日 访问 量 、 当 日 他 访问 量 进行 统计 ， 并 且 输 出 统计 结果 。 其 关键 代码 如 下 : 


| <td height="25" align="center" valign="middle"><?php 
| $query_1="select sum(counts) as countes from tb_count10 "; 
| /查询 数据 库 中 总 的 访问 量 
| Sresult_1=mysql query($query_1): 
| $countes_1=mysql result($result_1,0,countes); 
| echo "<p class='STYLE2> 网 站 总 的 访问 量 : $countes_1</p>"; 
| // 查 询 数据 库 中 总 的 中 访问 量 
$query_2="select * from tb_count10 "; 
$result 2=mysql query($query 2); 
while($myrow=mysql fetch array($result 2)){ 


$counts 2[]=$myrow[ip]; // 将 获取 的 他 的 值 赋 给 变量 
} 
$results_2=array_unique($counts 2); // 去 除数 组 中 重复 的 值 
$countes 2=count($results 2): // 获 取 数 组 中 值 的 数量 ， 即 总 的 正 访问 量 
echo "<p class= STYLE2> 网 站 总 的 卫 访问 量 : $countes 2</p>"; 


?></td> 
<td width="323" align="center" valign="middle"><?php 
/查询 数据 库 中 当月 总 的 访问 量 
$query_3="select sum(counts) as countes from tb count10 where data2=".substr(date("Y-m-d"),0,7)."™"; 
$result_3=mysql query($query_3); 
$countes_3=mysql_result($result_3,0,'countes'); 


2> 
<p class='STYLE2> 网 站 当月 的 访问 量 : <?php echo $countes_3;?></p> 
<2php 
/查询 数据 库 中 当月 的 下 访问 量 
$query_4="select * from tb_count10 where data2=".substr(date("Y-m-d"),0,7)."™"; 
$result_ 4=mysql query($query_4): 
| while($myrow_4=mysql fetch array($result 4)){ 
| S$counts_4[]=$myrow_4[ip]; /将 获取 的 王 的 值 赋 给 变量 
| } 
| $results_4=array_unique($counts 4); // 去 除数 组 中 重复 的 值 
| $countes 4=count($results 4): // 获 取 数 组 中 值 的 数量 ， 即 总 的 下 访问 量 
| echo "<p class=' STYLE2> 网 站 当月 下 的 访问 量 : $countes 4</p>"; 
| ?></td> 


<td width="294" align="center" valign="middle"><?php 
| // 查 询 数 据 库 中 当日 总 的 访问 量 
| $query_5="select sum(counts) as countes from tb_count10 where datal=".date("Y-m-d")." "; 
| Sresult 5-mysql_ query(Squery 5); 
S$countes 5=mysql_result($result_ 5,0,'countes'); 
| echo "<p class='STYLE2> 网 站 当日 的 访问 量 : $countes_5</p>"; 
| // 查 询 数据 库 中 当日 的 于 访问 量 
| $query_6="select * from tb_count10 where datal 一 ".date("Y-m-d")."": 
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$result 6=mysql query($query_6); 
while($myrow_6=mysql fetch array($result 6)){ 


$counts 6[]=$myrow 6[ip]: // 将 获取 的 他 的 值 赋 给 变量 

iflis_array($counts 6)){ 

$results 6=aray unique($counts 6); // 去 除数 组 中 重复 的 值 

$countes 6=count(Sresults 6); // 获 取 数 组 中 值 的 数量 ， 即 总 的 下 访问 量 

echo "<p class='STYLE2> 网 站 当日 人 P 的 访问 量 : $countes_6</p>"; 
} | 
?></td> | 
</t> | 


接着 ， 创 建 form 表单 ， 通 过 下 拉 列 表 将 指定 月 份 的 数据 提交 到 progress() 函 数 中 ， 完 成 对 | 
指定 月 份 网 站 访问 量 的 分 析 。 其 关键 代码 如 下 : 
<form name="forml" method="post"> 
<select name="selects" id="selects"> 
<option value="2008-01">2008-01</option> 
<option value="2008-02">2008-02</option> 
<option value="2008-03">2008-03</option> 
<option value="2008-04">2008-04</option> 
</select>&nbsp:;&nbsp; 
<input type="button" name="Submit" value=" 网 站 月 访问 量 分 析 " onClick="progress()"> 
</form> 


最 后 ， 创 建 mg 图 像 ，src 属性 指定 imgphp 文件 ， 该 文件 的 作用 是 通过 Jpgraph 生成 网 站 | 
月 访问 量 的 折线 分 析 图 。 
<td colspan="2" align="center" bgcolor="#FFFFFF"><img id="img" sre="img.php?m=0" /></td> | 

(3) 创建 img.php 文件 ， 通 过 Jpgraph 生成 网 站 月 访问 量 的 折线 分 析 图 。 这 里 根据 超 链 接 | 

中 传递 的 参数 m 的 值 ,执行 查询 操作 获取 数据 库 中 的 数据 ,获取 到 的 数据 就 是 折线 图 中 的 数据 。 | 
其 关键 代码 如 下 : | 


<2php 
require_once 'src/jipgraph.php'; // 导 入 Jpgraph 类 库 
require_once 'src/ipgraph line.php'; // 导 入 Jpgraph 类 库 的 柱状 图 功能 
include("conn/conn.php"); // 连 接 数 据 库 
$data=$_GET[m']; // 获 取 超 链接 传递 的 参数 
$query="select counts from tb_count10 where data2='$data' order by datal": | 
S$resultmysql_query($query); /执行 查询 操作 | 
while($myrow=mysql fetch array($result)){ | 
$dataTmp[]=current($myrow): /将 数组 中 的 当前 单元 添加 到 数组 中 | 
} | 
$datal = array(0, 0); /设置 统计 数据 | 
for($i=0: $i<count($dataTmp): $i++){ // 执 行 for 循环 | 
$datal[$i] = $dataTmp[$i]; /获取 数组 中 的 数据 | 
} | 
Sgraph = new Graph(800, 350); // 定 义 画 布 大 小 | 
S$eraph->SetScale("textlin"); /设置 刻度 样式 | 
Seraph->SetShadow0: /设置 背景 带 阴影 | 
Sgraph->img->SetMargin(50, 40. 40, 70): /设置 统计 图 边 距 范围 | 
$graph->title->Set("$data-- 网 站 流量 分 析 "): /设置 统计 图 标题 | 
"197 。 


人 Pp 六 须知 道 的 300 个 问题 


| Cs 

| $lineplotl = new LinePlot($datal): /建立 LinePlot 对 象 

| $graph->Add($lineplotl): 

| Sgraph->xaxis->title->Set(" 天 "); // 设 置 横 坐 标 名 称 

| Sgraph->yaxis->title->Set(" 网 站 流量 分 析 (个) ");， ”// 设 置 第 左 侧 纵 坐标 名 称 

| S$egraph->title->SetFont(FF_SIMSUN, FS_ BOLD): /设置 标题 字体 

| $egraph->yaxis->title->SetFont(FF_SIMSUN, FS BOLD); /设置 和 轴 字 体 
S$lineplotl1->SetColor('red"); // 设 置 折线 颜色 


$lineplot1->SetLegend($data."--"." 网 站 访问 流量 "); 

$graph->legend->SetFont(FF SIMSUN, FS_BOLD); 

$egraph->legend->SetLayout(LEGEND HOR); 

$egraph->legend->Pos(0.4, 0.96, 'center’, bottom ): 

$egraph->Stroke(); 

?> 
| (4) 在 根 目录 下 创建 js 文件 夹 ， 在 js 文件 夹 下 创建 funjs 文件 ， 并 将 JQuery 的 文件 包 复 
， 制 到 js 文件 夹 下 。 
| 在 funjs 文件 中 编写 自 定义 函数 , 通过 JQuery 属性 中 的 函数 实现 对 折线 图 中 数据 的 动态 加 
| 载 。 其 中 ，progress() 函 数 用 于 获取 指定 月 份 的 数据 ， 从 而 执行 beginProgress() 函 数 ， 完 成 对 指 
| 定 月 份 网 站 流量 的 分 析 ; 而 progres0) 函 数 用 于 获取 当前 月 份 的 数据 , 同时 也 执行 beginProgressO 
| 函数 , 对 当前 月 份 的 网 站 访问 量 进行 分 析 ; 在 beginProgress0 函 数 中 应 用 JQuery 中 的 attr 属性 ， 
| 设置 图 像 sre 的 属性 ， 实 现 折线 图 的 动态 更 新 。 其 代码 如 下 : 


| $graph->xaxis->title->SetFont(FF_SIMSUN, FS_BOLD): /设置 立轴 字体 


function progressO{ // 创 建 自 定义 函数 
counts=form!l .selects.value; // 获 取 表 单 下 拉 列 表 中 的 值 
beginProgress(counts): /将 获取 的 值 作为 参数 ， 执 行 自 定义 函数 beginProgress0 
] 
function progresO{ // 创 建 自 定义 函数 
var date_ time=new Date(): /创建 一 个 Date 对 象 


/定义 变量 ， 并 为 其 赋值 为 当前 年 份 ， 后 加 中 文 “ 年 ” 字 标 识 
Var date_times=getYear(Q+"-"; 
/获取 当前 月 份 。 注 意 月 份 从 0 开始 ， 所 以 需 加 1， 后 加 中 文 “ 月 ”标识 


date_times+=getMonthO+1; /在 JavaScript 脚本 中 获取 当前 时 间 
} 
beginProgress(date_ times); // 将 获取 的 时 间作 为 参数 ， 执 行 自 定义 函数 beginProgress 
| setTimeout("progresO", 60000); /间隔 一 段 时 间 自 动 执行 一 次 progres 函数 
| } 
| function beginProgress(counts){ // 创 建 自 定义 函数 
| $("#img").attr("sre", "img.php?m="+counts); // 为 图 像 设置 src 属性 
| } 


问题 150 如何 通过 Jpgraph 类 库 输 出 中 文字 符 事 ? 


问题 讲述 
Jpgraph 类 库 是 一 个 可 以 应 用 在 PHP4.3.1 以 上 版 本 的 用 于 图 形 图 像 绘 制 的 类 库 ， 该 类 库 完 
全 基于 GD2 函数 库 编号， 提供 了 多 种 方法 创建 各 类 统计 图 ， 包 括 坐 标 图 、 柱 状 图 、 饼 形 图 等 。 
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由 于 了 pgraph 类 库 在 编写 时 采用 的 是 UTF-8 编码 ， 所 以 它 在 输出 中 文字 符 串 时 会 出 现 乱 码 的 问 | 
题 ， 对 此 虽然 网 络 上 有 很 多 的 解决 方案 ， 但 是 效果 一 般 。 因 此 ， 我 们 在 这 里 整理 出 一 个 有 效 的 | 
解决 方案 ， 供 读者 参考 和 学 习 。 
专家 解答 
解决 了 graph 类 库 中 输出 中 文字 符 串 乱码 的 问题 ， 可 以 分 3 步 走 : 
第 一 步 : 创建 编码 格式 为 UTF-8 的 网 页 文件 。 
第 二 步 : 通过 iconv0 函 数 实现 中 文字 符 串 编码 格式 的 转换 , 将 UTF-8 编码 转换 为 gb2312 | 
第 三 步 : 设置 中 文字 符 串 的 字体 (简体 为 FF_SIMSUN, FS_BOLD, 繁体 为 FF_CHINESE)。| 
通过 上 述 3 步 即 可 圆满 地 解决 pgraph 类 库 中 输出 中 文字 符 串 乱码 的 问题 。 | 
注意 : 
此 处 应 用 的 Jpgraph 的 版 本 是 3.0.7。 


应 用 示例 


应 用 Jpgraph 类 库 创 建 一 个 3D 饼 形 图 , 输出 MySQL 图 书 各 部 分 内 容 所 占 比例 。 创建 一 个 | 
UTF-8 编码 格式 的 index.php 文件 ， 载 入 Jpgraph 类 库 文件 jpgraphphp、jpgraph pie.php 和 | 
jpgraph_pie3d.php， 调 用 类 库 中 的 方法 ， 完 成 3D 饼 形 图 的 设计 。 其 关键 代码 如 下 : 

<2php 

header ( "Content-type: text/html: charset=UTF-8" ): /设置 文件 编码 格式 
include ("../src/jpgraph.php"); 

include ("../src/jpgraph_pie.php"); 

include ("../src/jpgraph_pie3d.php"); 


$data = atray(8.5.4.3); /定义 数据 大 小 
$graph = new PieGraph(750.450,"auto"); // 创 建 图 像 
$graph->SetShadow0: /创建 图 像 阴影 


$egraph->tabtitle->Set(iconv("utf-8","gb2312","MySQL 图 书 内 容 分 布 比例 ") ); | 
/输出 标题 ， 进 行 编 码 格式 转换 | 


Seraph->tabtitle->SetFont(FF_SIMSUN, FS_BOLD.14): // 设 置 标题 字体 
S$graph->title->SetColor("darkblue"); // 定 义 标题 颜色 
$egraph->legend->SetFont(FF_SIMSUN, FS _ BOLD,14); // 设 置 辅助 标题 字体 
S$graph->legend->Pos(0.03,0.1): // 控 制 注释 文字 的 位 置 

$pl = new PiePlot3d($data); /创建 图 像 | 
$p1->SetTheme("sand'"): // 控 制图 像 颜色 | 
S$p1->SetCenter(0.35): /设置 图 像 位置 | 
$p1->SetSize(0.4); /设置 图 像 大 小 | 
S$p1->SetHeight(40): // 设 置 饼 形 图 高 度 | 
$p1->SetAngle(45): /设置 图 像 倾 斜 角度 | 
$p1->Explode(array(32,20.16,18)): // 控 制 饼 形 图 的 分 割 | 
$p1->value->SetFont(FF_ ARIAL.FS NORMAL.10): /设置 内 容 字体 和 字号 大 小 


$p1->SetLegends(array(iconv("utf-8","gb2312","MySQL 基础 篇 "),iconv("utf-8","gb2312", "MySQL | 
高 级 应 用 篇 "),iconv("utf-8","gb2312","MySQL 系统 管理 篇 ").iconv("utf-8","gb2312", "MySQL 应 用 程序 开发 | 


篇 "))); // 输 出 辅助 标题 内 容 
$graph->Add($p]): /添加 数据 
S$graph->Stroke0: // 生 成 图 像 
?> 
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运行 结果 如 图 6.9 所 示 。 


MySQL 图 书 内 容 分 布 比例 日 MySQL 基 础 篇 
口 My5QL 高 级 应 用 篇 
口 MySQL 系 统管 理 篇 
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图 6.9 输出 中 文字 符 串 


专家 点 评 

通过 此 方法 在 Jpgraph 类 库 中 输出 中 文 适用 于 Jpgraph 类 库 的 各 系列 版 本 。 另 外 ， 如 果 用 
户 通过 XAMPP 集成 化 安装 包 配 置 的 PHP 开发 环境 , 那么 在 应 用 此 方法 输出 中 文 时 有 可 能 也 会 
出 现 乱 码 ， 因 为 它 在 处 理 GD 库 方面 存在 一 些 问 题 ， 至 于 使 用 其 他 的 集成 化 安装 包 或 手动 配置 
的 PHP 开发 环境 ， 使 用 上 述 方法 都 可 以 完成 中 文 的 正常 输出 。 


问题 151] 如何 制作 数字 图 像 验 证 码 ? 


| 问题 阐述 


为 了 防止 用 户 通过 恶意 程序 登录 站 点 ， 提 高 网 站 的 安全 性 ， 项 目 开 发 时 ， 在 用 户 登录 、 注 


| 册 及 发 表 主题 等 模块 中 加 入 验证 码 模块 是 必 不 可 少 的 。 应 用 验证 码 技术 可 以 有 效 地 防止 用 户 通 


过 探测 的 方式 非法 登录 ， 从 而 有 效 地 提高 站 点 的 安全 性 。 
专家 解答 


制作 数字 图 像 验证 码 , 首先 , 通过 JavaScript 中 Math 对 象 的 random() 方 法 随机 生成 4 位 随 
机 数 ， 并 通过 doument 对 象 的 write( 方 法 将 这 4 位 随机 数 所 对 应 的 图 片 以 链接 形式 输出 。 
然后 ， 通 过 $_GET[] 全 局 数组 提取 链接 地 址 所 传 入 的 随机 数值 ， 通 过 PHP 文件 操作 函数 读 
取 该 随机 数 对 应 的 图 片 ， 并 以 二 进 制 的 形式 输出 到 浏览 器 。 
接着 ， 定 义 一 个 重新 指定 图 片 链接 地 址 的 方法 ， 当 单 击 “ 看 不 清 ” 超 链接 时 ， 调 用 该 方法 
生成 新 的 验证 码 图 片 。 
最 后 ， 在 表单 中 添加 一 个 隐藏 域 ， 用 于 提交 4 位 随机 数 ， 当 用 户 单 击 表单 中 的 “登录 ” 按 
钮 时 ， 通 过 JavaScript 将 隐藏 域 的 值 与 验证 码 文本 框 所 输入 的 内 容 进行 比较 ， 完 成 验证 操作 。 
(1) 创建 4 位 随机 数 
通过 JavaScript 中 Math 对 象 的 random() 方 法 随机 生成 4 位 随机 数 。 关 键 代码 如 下 : 
var num1=Math.round(Math.random()*10000000); 
Var num=num1 :toString().substr(0,4); // 生 成 4 位 随机 数 
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(2) 将 图 片 以 二 进 制 形式 输出 | 

应 用 PHP 的 文件 操作 函数 即 可 实现 将 图 片 以 二 进 制 的 形式 输出 到 浏览 器 ,应 用 fiead0 画 数 
实现 ， 其 语法 如 下 : | 
fread ( int handle, int length) 


参数 说 明 : 

@ handle: fopen() 函 数 打 开 文 件 后 所 返回 的 文件 句柄 。 

@ length: 所 读 取 文 件 的 长 度 。 

例如 ， 本 实例 中 应 用 fread0 方 法 将 图 片 以 二 进 制 形 式 输出 到 浏览 器 ， 实 现代 码 如 下 : 


$fp=fopen($address,"r"); /打开 文件 
echo fread($fp,filesize($address)); // 读 取 文 件 
应 用 示例 


在 本 示例 中 ,应 用 JavaScript 技术 ， 再 以 0 到 9 这 10 个 别致 数字 图 片 为 辅助 ， 制 作 博 客 后 
台 登 录 的 验证 码 。 其 具体 步骤 如 下 : | 
(1) 编写 index.php 用 户 登录 页 面 ， 创 建 表 单 ， 在 表单 中 添加 用 户 名 文本 域 、 密 码 域 、 验 | 
证 码 文本 域 以 及 保存 随机 数 的 隐藏 域 。 | 
(2) 创建 fnn.js 脚本 文件 ， 在 客户 端 对 表单 提交 数据 进行 验证 。 | 
(3) 编写 JavaScript 脚本 生成 4 位 随机 数 ， 并 且 在 用 户 单 击 “ 看 不 清 ” 超 链接 时 ， 调 用 自 | 
定义 方法 code_10 动 态 更 改 验 证 码 的 值 。 其 关键 代码 如 下 : | 
<script language="javascript"> 
var num1=Math.round(Math.random()*10000000); 
var num=numl.toStringO.substr(0.4): /产生 4 位 随机 数 
document.write("<img name=codeimg4 stre='xym.php?num="+num.substr(0,1)+">"); 
document.write("<img name=codeimg5 sre='xym.php?num="+num.substr(1,1)+">"); 
document.write("<img name=codeimg6 ste='xym.php?num="+num.substr(2,1)+">"); 
document.write("<img name=codeimg7 ste='xym.php?num="+num.substr(3,1)+">"); 
form login.xyml .value=num:; 


function code_10{ // 通 过 该 方法 动态 改变 验证 码 的 值 
var num1=Math.round(Math.random()*10000000); 
Var num=numl.toStringO.substr(0.4); 


document.codeimg4.sre="xym.php?num="+num.substr(0,1); 

document.codeimg5.sre="xym.php?num="+num.substr(1,1); 

document.codeimg6.sre="xym.php?num="+num.substr(2,1); 

document.codeimg7.sre="xym.php?num="+num.substr(3,1); 

form login.xyml.value=num: Re 
) 


</script> 
(4) 创建 xym.php 文件 ， 采 用 PHP 文件 操作 函数 将 图 片 以 二 进 制 的 形式 输出 到 浏览 器 ， 
这 样 可 以 有 效 地 提高 静态 页 面 与 动态 页 面 之 间 的 交互 性 。 其 代码 如 下 : 


$num=$_GET[num]; /接收 Get 方法 传递 过 来 的 参数 
$address="images/code/".$num "gif' /获取 图 片 路 径 
$fp=fopen($address,"r"); /打开 图 片 文 件 

echo fread($fp.filesize($address)); // 将 图 片 文件 读 取 到 浏览 器 
felose($fp); /关闭 打开 的 文件 句柄 
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运行 本 示例 ， 其 运行 结果 如 图 6.10 所 示 。 


ARPES:FF :124 下 下 下 
5: [869999 | | 


图 6.10 数字 图 像 验 证 码 


专家 点 评 


如 果 在 PHP 页 面 中 不 仅 包含 PHP 代码 ， 而 且 包 含 其 他 语言 代码 ， 如 JavaScript、CSS 或 
HTML 等 ， 则 需要 写 完整 的 PHP 代码 块 标记 ， 即 “<?php” 与 “?>” 都 要 写 , 如 果 PHP 页 面 代 
| 码 完 全 由 PHP 构成 ， 则 建议 只 写 开始 标记 ， 不 写 结束 标记 ， 这 样 可 以 防止 恶意 用 户 通过 非法 
注入 等 手段 在 页 面 代 码 后 注入 恶意 代码 。 


问题 152 ”如 何 制 作 中 文 图 像 验证 码 ? 


问题 前述 


验证 码 不 但 可 以 使 用 数字 图 像 来 完成 ， 而 且 可 以 使 用 中 文 图 像 来 完成 。 那 么 如 何 制作 中 文 
图 像 验 证 码 呢 ? 


| 专家 解答 


与 开发 数字 验证 码 不 同 ， 开 发 中 文 图 像 验 证 码 ， 首 先 应 确定 所 使 用 的 文字 ， 并 且 将 文字 制 
作成 图 片 ， 以 数字 进行 命名 。 然 后 ， 将 所 使 用 的 文字 定义 到 数组 中 。 最 后 ， 计 算出 数组 中 元 素 
| 的 长 度 值 ， 并 应 用 rand0 函 数 从 这 个 长 度 值 中 随机 取 值 ， 根 据 随机 取 值 输出 对 应 的 中 文 图 像 ， 
即 验证 码 。 


应 用 示例 


在 本 示例 中 ， 制 作 博客 后 台 登 录 页 面 ， 并 采用 中 文 汉字 生成 验证 码 。 其 具体 步骤 如 下 : 

| (1) 编写 index.php 用 户 登 录 页 面 ， 创 建 表单 ， 在 表单 中 添加 用 户 名 文本 域 、 密 码 域 、 验 
| 证 码 文本 域 以 及 保存 随机 数 的 隐藏 域 。 

| (2) 在 index.php 页 面 中 , 编写 PHP 脚本 ,定义 验证 码 所 使 用 的 中 文字 符 串 ， 并 生成 验证 
| 码 ， 其 关键 代码 如 下 : 


Sstr= array (大 " "更 " " 创 " "天 " "各"、" 客 "," 博 "，" 技 "," 立 ", "新 "): // 定 义 中 文字 符 串 的 数组 
a // 计 算出 数组 中 元 素 个 数 
for($i=0; $1<4; $i1++) { /执行 for 循环 ， 以 数组 中 元 素 个 数 为 范围 随机 取出 4 个 值 
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$num = rand ( 0, $word - 1 ); $word=$word*2-1 /随机 取 值 
/输出 随机 取出 的 4 个 值 对 应 的 中 文 图 片 

$img = $img . "<img src=' images/checkcode/" . $num . ".gif width 一 16 height="16>"; | 

/显示 随机 图 片 | 

S$pic = $pic . $str [$num]; // 将 图 片 转换 成 数组 中 的 文字 | # 
} | 估 - 
,二 

te 
(4) 创建 checkuserphp 文件 ， 完 成 对 用 户 提交 的 登录 信息 的 验证 。 


博客 后 台 登 录 页 面 的 运行 结果 如 图 6.11 所 示 。 


ps; 了 am: RE | 
“mb 全 下 | 


图 6.11 中 文 图 像 验 证 码 


专家 点 评 


在 本 示例 中 ， 应 用 count(O) 函 数 计算 数组 中 元 素 的 数目 。 注 意 ，countO 函 数 只 适用 于 计算 数 
组 中 元 素 的 个 数 ， 如 果 将 其 应 用 到 其 他 类 型 的 数据 中 ， 那 么 返回 值 只 有 一 个 单元 。 如 果 要 计算 
字符 串 的 长 度 ， 那 么 要 使 用 stlen0 函 数 ， 切 记 strlen0) 函 数 不 可 对 数组 元 素 进 行 操作 ， 虽 然 不 返 
回 错误 信息 ， 但 是 返回 的 结果 也 是 不 准确 的 。 | 


问题 153 如何 实现 验证 码 在 当前 页 验证 ? | 


问题 讲述 


验证 码 技术 是 为 了 防止 填写 的 信息 被 暴力 破解 ， 在 签 写 信息 时 ， 通 常 被 要 求 输入 验证 码 。 通 
常情 况 下 ， 如 果 输 入 的 验证 码 错误 ， 将 重新 刷新 页 面 ， 所 签 写 的 信息 将 被 清空 ， 这 样 给 用 户 操作 | 
造成 了 一 定 的 麻烦 。 为 了 避免 这 一 现象 的 发 生 , 就 需 实现 让 验证 码 在 当前 页 验证 , 该 如 何 实现 ? | 
专家 解答 | 

实现 验证 码 在 当前 页 验证 可 以 通过 JavaScript 脚本 来 实现 ， 封 装 自 定义 函数 check_form()， | 
获取 表单 提交 的 验证 码 ， 实 现 对 验证 码 的 验证 。 其 代码 如 下 : | 

<script language="javascript"> 
function check form(form1){ 
if(forml.checkcode.value 一 ""){ /如 果 验 证 码 为 空 


alert(" 验 证 码 不 能 为 空 ! ");forml.checkcode.focus():return false; 
} 
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if(forml.checkcode.value!=num){ /如果 输 入 的 验证 码 的 值 与 随机 的 验证 码 的 值 不 相等 
alert(" 您 输入 的 验证 码 不 正确 ， 重 新 输入 ! "):forml.checkcode .focusO:retum false; 
} 


问题 154 ”如 何 任 意 调整 图 片 的 大 小 ? 


问题 阐述 

一 般 情况 下 ， 我们 上 传 的 图 片 都 是 需要 手动 修改 统一 大 小 ， 即 使 可 以 通过 缩 略 图 的 方式 统 
一 控制 上 传 图 片 的 大 小 ， 但 是 有 些 时 候 不 是 很 理想 。 那 么 是 否 有 办 法 可 以 任意 在 程序 中 调整 图 
片 的 大 小 呢 ? 
专家 解答 

任意 调整 上 传 图 片 大 小 方法 实现 的 原理 : 

首先 ， 根 据 表 单 中 提交 的 值 获取 指定 图 片 的 存储 路 径 和 名 称 以 及 它 要 被 调整 的 比例 。 

然后 ， 应 用 getimagesize() 函 数 获取 指定 图 片 的 数据 。 

接着 ， 定 义 thumb() 方 法 ， 根 据 原始 图 片 的 数据 和 提交 的 调整 比例 数据 ， 判 断 是 根据 比例 
缩放 还 是 进行 裁剪。 

最 后 ， 载 入 原始 图 片 ， 并 定义 新 的 缩 略 图 ， 应 用 ImageCopyResampled0) 函 数 将 原始 图 像 复 
制 到 新 的 缩 略 图 中 ， 并 将 新 的 图 像 存储 到 指定 位 置 。 

imagecopyresampled() 函 数 ， 从 原 图 像 中 采样 ， 复 制 部 分 图 像 ， 重 新 生成 图 像 。 

JImagecopyresized0 函 数 ， 同 样 实现 图 像 的 复制 操作 。 语 法 如 下 : 

bool imagecopyresized ( resource dst_image, resource SIc_image, int dst_x, int dst_y, int src_X， int 

sIC_y, int dst_w, int dst_h, int sre_w, int src h ) 

imagecopyresized() 将 一 幅 图 像 中 的 一 块 正方 形 区 域 复 制 到 另 一 个 图 像 中 。 

参数 说 明 : 

@ dst image: 是 目标 图 像 标 识 符 。 

@ src image: 是 源 图 像 标 识 符 。 

如 果 源 图 像 和 目标 图 像 的 宽度 和 高 度 不 同 ， 则 会 进行 相应 的 图 像 收 缩 和 拉 伸 。 坐 标 指 的 是 
左上 角 。 本 函数 可 用 来 在 同一 幅 图 内 部 复制 (如果 dst image 和 src_image 相同 ) 区 域 ， 但 如 果 
区 域 交 迭 则 结果 不 可 预知 。 


说 明 : 

因为 调 色 板 图 像 限制 (255+1 种 颜色 ) 有 个 问题 。 重 新 采样 或 过 滤 图 像 通常 需要 多 于 255 
种 颜色 ， 计 算 新 的 被 重新 采样 的 像素 及 其 颜色 时 采用 了 一 种 近似 值 。 对 调 色 板 图 像 尝试 分 配 一 
个 新 颜色 时 ， 如 果 失 败 那么 选择 计算 结果 最 接近 (理论 上 ) 的 颜色 ， 这 并 不 是 视觉 上 最 接近 的 
颜色 。 这 可 能 会 产生 怪异 的 结果 ， 如 空白 (或 者 视觉 上 是 空白 ) 的 图 像 。 要 跳 过 这 个 问题 ， 应 
使 用 真 彩色 图 像 作为 目标 图 像 ， 如 使 用 imagecreatetruecolor0) 函 数 创 建 的 图 像 。 


应 用 示例 
在 本 示例 中 ， 开 发 一 个 图 片上 传 的 功能 ， 并 且 可 以 对 上 传 图 片 大 小 任意 调整 。 其 具体 步骤 
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如 下 : 
(1) 创建 一 个 图 片上 传 功能 ， 将 图 片上 传 到 服务 器 指定 的 文件 夹 下 ， 并 通过 文件 系统 函 
数 读 取出 存储 在 指定 目录 下 的 图 片 。 


尺寸 ， 将 数据 提交 到 index_ok.php 文件 中 。 


(3) 创建 index_ok php 文件 ， 包 含 thumbnailphp 文件 ， 实 例 化 image 类 ， 调 用 其 中 的 | 


thumb() 方 法 ， 将 表单 中 提交 的 数据 作为 参数 ， 完 成 对 图 片 大 小 的 调整 。 其 关键 代码 如 下 : 


<2?php 

这 $_ POST['Submit] 一 "调整 图 像 大 小 7){ 

include("thumbnail.php"); // 包 含 图 像 类 存储 文件 
Simg = new image(); // 实 例 化 图 像 处 理 类 


$return=$img->param($ POST['pic_path"])->thumb("./thumbnail/".$_ POST[pic_ name], $ POST[pic width], 
$_ POST[pic_ height],1); 

echo "<script>alert( 图 像 大 小 设置 成 功 ! ); window.location.href='index.php';</script>"; 

上 


> 


(4) 创建 humbnail.php 文件 , 编写 image 类 , 定义 thumb() 方 法 , 根据 表单 中 提交 的 数据 ， 
完成 图 像 大 小 的 调整 操作 。 其 关键 代码 如 下 : 


//thumb( 新 图 地 址 ， 宽 ,高 , 裁剪， 允许 放大 ) 
public function thumb($filename, Snews_w = 100, $news_h = 100, $cut = 0, $big = 0) { 
S$info = $this->getImageInfo ( $this->img ); // 获 取 原 图 信息 
if(! empty ( $info [0] ) { 
S$old_w = $info [0]; 
$old_h = $info [1]; 
Snew_w = $old w* $news_w: 
Snew_h= $old h * $news_h; 


$type = $info ['type']; 
$ext = $info [ext]:; 
unset ( $info ): 
过 ($old w < $new_h && $old h<$new_ w&&lS$big){ ”// 如 果 原 图 比 缩 略 图 小 
并 且 不 允许 放大 
return false: 
} 
if ($cut—1){ // 居 中 裁剪 


$scalel = round ( $new_w / $new_h, 2 ); 
$scale2 = round ( $old_w / $old h, 2); 
if ($scalel > $scale2) { 
S$end h= round ( $old w/ $scalel, 2 ); 
$start h= (gold h - $end h) /2: 
Sstart w= 0; 
$end w= $old w: 
} else { 
S$end w= round ( $old h* $scalel,2); 
$start w= ($old w - $end_w) /2; 
S$start h=0; 
$end h= $old h: 
} 
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(2) 在 图 片 输出 的 同时 ， 在 右 侧 创建 一 个 表单 ， 提 交 图 片 的 存储 位 置 、 名 称 和 要 调整 的 | 
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CS 
S$width = $new_w:; // 定 义 图 像 宽 
Sheight = $new_h; // 定 义 图 像 高 
} 
S$createFun = 'TmageCreateFrom' . $type; // 新 建 图 像 
S$oldimg = $createFun ( $this->imsg ); 
if ($type (= 'gif && function exists ('imagecreatetruecolor )) { 
$newimg = imagecreatetruecolor ( $width, $height ); /创建 新 的 真 彩 图 像 
}else{ 
Snewimg = imagecreate ( $width, $height ); // 创 建新 的 图 像 
} 
if (function exists ( "ImageCopyResampled" )) { 
ImageCopyResampled ( $newimg, $oldimg, 0, 0, $start w, $start h, $width, 
Sheight, $end_w, $end h ); // 完 成 图 像 的 复制 
} else { 
JmageCopyResized ( $newimg, $oldimg, 0, 0, $start_w, $start_h, $width, $height, 
S$end_w, $end h ); 
}; 
$type — 'jpeg && imageinterlace ( $newimg, 1 ); // 对 jpeg 图 形 设置 隔行 扫描 
SimageFun = 'image' . $type; /生成 图 片 
!1@$imageFun ( $newimg, $filename ) && die (' 保 存 失败 !); 
ImageDestroy ( $newimg ); /销毁 图 像 
retum $filename; // 返 回 图 像 名 称 


} 
} 


运行 本 示例 ， 在 上 传 图 片 的 右 侧 选择 要 调整 图 片 大 小 的 比例 ， 然 后 单 击 下 面 的 按钮 ， 其 运 
行 结 果 如 图 6.12 所 示 。 


图 6.12 ”任意 调整 图 像 的 大 小 
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吾 吾 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 至 


问题 155 
问题 156 
问题 157 
问题 158 
问题 159 
问题 160 
问题 161 
问题 162 
问题 163 
问题 164 
问题 165 
问题 166 
问题 167 
问题 168 
问题 169 
问题 170 
问题 171 
问题 172 
问题 173 
问题 174 


如 何 限制 上 传 文件 的 大 小 ? 

如 何 判断 上 传 文件 是 否 符合 要 求 ? 

如 何 通过 header(O 函 数 进行 下 载 ? 
如 何 实现 多 文件 同时 上 传 ? 

如 何 判 断 文件 的 权限 ? 

如 何 获取 磁盘 分 区 的 大 小 ? 

如 何 遍历 指定 目录 下 的 所 有 文件 ? 

如 何 应 用 fopen0O 函 数 的 选择 模式 ? 

如 何 锁定 文件 ? 

如 何 访问 远程 文件 ? 

如 何 读 取 本 地 文件 ? 

如 何 使 用 文件 句柄? 

如 何在 文件 中 存储 配置 信息 ? 

如 何在 PHP 中 创建 压缩 的 RAR 文件 ? 
如 何在 PHP 中 对 RAR 文件 进行 解压 ? 
如 何 将 文本 文件 中 的 数据 存储 到 数据 库 中 ? 
如 何 判断 一 个 远程 的 文件 是 否 存 在 ? 

如 何 将 上 传 文件 转换 成 RAR 文件 ? 

如 何 遍历 、 删 除 指定 目录 下 的 所 有 文件 ? 
如 何 使 用 文本 文件 统计 网 站 访问 量 ? 


诬 
需 
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问题 1355 ”如何 限制 上 传 文件 的 大 小 ? 


网 -问题 阐述 


在 网 站 开发 的 过 程 中 ， 为 了 确保 能 够 充分 利用 服务 器 的 空间 ， 在 开发 上 传 功能 时 ， 必 须 对 
| 上 传 文件 的 大 小 进行 控制 。 那 么 如 何 对 上 传 文件 的 大 小 进行 控制 呢 ? 


| 专家 解答 


| 控制 上 传 文件 的 大 小 可 以 从 两 个 方面 入 手 : 第 一 是 在 PHP 的 配置 文件 php.ini 中 对 上 传 文 
| 件 进行 控制 ， 如 果 上 传 文件 超过 它 指定 的 范围 ， 那 么 执行 上 传 就 会 失败 ; 第 二 是 在 PHP 配置 
| 文件 允许 的 范围 内 ， 在 程序 中 对 上 传 文件 的 大 小 进行 控制 。 
| 1. 通过 配置 文件 控制 上 传 文件 
| PHP 中 通过 php.ini 文件 对 上 传 文件 进行 控制 ， 包 括 是 否 支 持 上 传 、 上 传 文件 的 临时 目录 、 
| 上 传 文件 的 大 小 、 指 令 执 行 的 时 间 、 指 令 分 配 的 内 存 空 间 。 
| 在 php.ini 中 ， 定 义 到 File Uploads 项 ， 完 成 对 上 传 相关 选项 的 设置 。 上 传 相关 选项 的 含义 
| 如下: 
| file_uploads: 如 果 值 是 on， 说 明 服 务 器 支持 文件 上 传 ; 如 果 为 off， 则 不 支持 。 一 般 
默认 是 支持 的 ， 这 个 不 用 修改 。 
upload tmp _ dir， 上 传 文件 临时 目录 。 在 文件 被 成 功 上 传 之 前 ， 文 件 首先 存放 到 服务 
器 端的 临时 目录 中 。 多 数 使 用 系统 默认 目录 ， 但 是 也 可 以 自行 设置 。 
upload max filesize: 服务 器 允许 上 传 文件 的 最 大 值 , 以 MB 为 单位 .系统 默认 为 2MB， 
如 果 网 站 需要 上 传 超过 2MB 的 数据 ， 那 么 就 要 修改 这 个 值 。 
| 上 述 是 php.ini 中 File_Uploads 项 中 与 上 传 相关 选项 参数 设置 说 明 , 除了 File_Uploads 项 中 
| 的 内 容 外 ， 在 php.ini 中 还 有 其 他 几 个 选项 会 影响 到 文件 的 上 传 。 
| 回 “max_execution_ time: PHP 中 一 个 指令 所 能 执行 的 最 大 时 间 ， 单 位 是 秒 。 该 选项 在 上 
传 超大 文件 时 必须 要 修改 ， 和 否则 即使 上传 文件 在 服务 器 允许 的 范围 内 ， 但 是 若 超过 
了 指令 所 能 执行 的 最 大 时 间 ， 仍 然 无 法 实现 上 传 。 
memory limit: PHP 中 一 个 指令 所 分 配 的 内 存 空间 ， 单 位 是 MB。 它 的 大 小 同样 会 影 
响 到 超大 文件 的 上 传 。 
2. 在 程序 中 控制 上 传 文件 


| 在 客户 端 控 制 上 传 文件 应 用 的 是 form 表单 中 的 enctype 和 method 属性 以 及 隐藏 域 
| MAX FILE SIZE。 

| enctype="multipart/form-data": 指定 表单 编码 数据 方式 。 

method="post": 指定 数据 的 传输 方式 。 

<input type="hidden" name="MAX FILE SIZE" value="10000" />: 通过 隐藏 域 控 制 上 
传 文件 的 大 小 (单位 为 字 节 ) , 该 值 不 能 超过 php.ini 配置 文件 中 upload_ max filesize 
选项 设置 的 值 。 它 不 能 完全 控制 上 传 文件 的 大 小 ， 只 是 可 以 避免 一 些 不 必要 的 麻烦 。 


四 办 罗 ' 
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技巧 : 
在 应 用 隐藏 域 <input type="hidden" name="MAX FILE SIZE" value="10000" /> 控制 上 传 文 
件 大 小 时 ， 必 须 将 其 放置 在 文件 域 之 前 ， 否 则 是 不 能 发 挥 作用 的 。 


3. 通过 全 局 变量 $ FILES 获取 上 传 信息 


上 述 两 种 判断 的 结果 都 可 以 通过 全 局 变量 $_FILES 的 返回 值 来 体现 .$_FILES 是 一 个 数组 ， | 


它 包 含 所 有 上 传 的 文件 信息 。 下 面 介绍 $_FILES 数组 中 每 个 元 素 的 含义 ， 如 表 7.1 所 示 。 


表 7.1 $_FILES 数组 中 的 元 素 含义 


说 明 
存储 上 传 文件 的 文件 名 ， 如 texttxt、title jpg 等 
存储 文件 大 小 。 单 位 为 字 节 
存储 文件 在 临时 目录 中 使 用 的 文件 名 。 因 为 文件 在 上 传 时 ， 首 先 要 将 其 
以 临时 文件 的 身份 保存 在 临时 目录 中 
存储 上 传 文件 的 MIME 类 型 ， 该 类 型 规定 各 种 文件 格式 的 类 型 。 每 种 
MIME 类 型 都 是 由 “/” 分 隔 的 主 类 型 和 子 类 型 组 成 的 ， 如 “image/gif”， 
主 类 型 为 “图 像 ”， 子 类 型 为 GIF 格式 的 文件 ，“texthtml” 代 表 文 本 的 
HTMI 文 件 
存储 与 文件 上 传 相关 的 错误 代码 。 此 项 目 是 PHP 4.2.0 版 本 中 新 增 内 容 。 
其 返回 值 有 5 种 ， 如 下 所 示 : 

0: 表示 没有 任何 错误 ， 文 件 上 传 成 功 

1: 表示 上 传 文件 的 大 小 超出 了 PHP 配 置 文件 指令 upload_max_filesize 选 
项 限制 的 值 

2: 表示 上 传 文件 大 小 超出 了 HTML 表 单 中 MAX_FILE_SIZE 选 项 所 指定 
的 值 

3: 表示 文件 只 被 上 传 了 一 部 分 

4: 表示 没有 上 传 任何 文件 


$_FILES[filenamej[size 


$_FILES[filename][tmp_name] 


$_FILES[filenamel][type] 


$_FILES[filename][error] 


4. 应 用 示例 
本 示例 实现 一 个 文件 上 传 功能 ， 并 且 控 制 上 传 文件 大 小 在 2MB 之 内 。 如 果 超 出 大 小 ， 则 
返回 相应 的 错误 信息 。 其 具体 步 又 如 下 : 
(1) 创建 index.php 文件 。 添加 表单 ， 设 置 隐藏 域 、 文 件 域 、 提 交 按 钮 ， 使 用 POST 方法 ， 
设置 enctype="multipart/form-data"， 将 数据 提交 到 本 页 。 
(2) 通过 $ FILES 获取 上 传 文件 的 相关 信息 ,通过 move_upload_file() 函 数 完成 图 片上 传 。 
代码 如 下 : 
<2php 
这 !empty($_FILES[up_picture][name])){ /判断 上 传 内 容 是 否 为 空 
这 $_FILES[up_picture]['emror]>0){ // 判 断 文 件 是 否 可 以 上 传 到 服务 器 
echo "上 传 错误 :"; 
Switch($ FILES[up_picture]['error]){ 
case 1: 
echo "上 传 文件 大 小 超出 配置 文件 规定 值 "; 
break; 
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Case 2: 
echo "上 传 文件 大 小 超出 表单 中 约定 值 "; 
break; 


Case 3: 
echo "上 传 文件 不 全 "; 
break; 


case 4: 
ee echo "没有 上 传 文件 


break; 
jelse{ 


if(!lis_dir("./upfile/")){ // 判 断 指定 目录 是 否 存 在 
mkdir("./upfile/"); // 创 建 目录 


$path="./upfile/'.time().strstr($_FILES["up_picture"]['name'],…");// 定 义 文件 名 称 和 存储 位 置 
ifis uploaded file($_ FILES[up_picture']['tmp_ name'])){ ”// 是 否 是 HTTP POST 上 传 
这 !move uploaded file($_FILES[mp picture]['tmp_name'].$pathb){ /执行 上 传 


echo "上 传 失 败 "; 
}else{ 
| echo "文件 ".time0.$_ FILES[mp_picture][mame']." 上 传 成 功 , 大 小 为 : ".$_FILES 
| [up_picture']['size']; 
| b 
}else{ 
echo "上 传 文件 ".$_FILES[up_pictute'][mame']." 不 合法 ! "; 
) 
} 
} 
7 


其 运行 结果 如 图 7.1 所 示 。 


© 一 个 点 起 _mT 人 Ne 
一 -一 ed 


选择 上 传 文件 : Fend_Franevork_IH cha 攻克 


上 传 图 片 大 小 为 (20) 


上 传媒 误 : 上 传 文件 大 小 超出 配置 文件 规定 值 


图 7.1 限制 上 传 文件 的 大 小 


| 
| 
| 
| 
| 
| Copyright © 1999-2010 吉林 省 明日 科技 有 限 公司 
| 
| 
| 
| 
| 


| 如 果 要 上 传 超大 的 文件 ， 则 必须 修改 php.ini 文件 。 修 改选 项 包括 upload_max_filesize 的 最 
| 大 值 、max_execution time 一 个 指令 所 能 执行 的 最 大 时 间 和 memory_limit 一 个 指令 所 分 配 的 内 
| 存 空间 ,否则 即使 上 传 文件 大 小 在 规定 的 范围 之 内 ， 由 于 指令 的 执行 时 间 不 足 或 者 指令 分 配 的 
内 存 不 足 都 无 法 顺利 地 完成 文件 的 上 传 操作 。 
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问题 156 ”如 何 判 断 上 传 文件 是 否 符 合 要 求 ? 


问题 痢 述 | 鲜 


文件 上 传 功 能 几乎 是 所 有 网 站 所 必 备 的 ， 然 而 文件 上 传 对 于 服务 器 来 说 具有 很 大 风险 ， 因 
此 对 文件 大 小 和 文件 类 型 进行 限制 就 显得 十 分 有 必要 。 控制 上 传 文件 大 小 的 问题 在 前 面 内 容 中 | 
已 经 解决 ， 那 么 如 何 对 上 传 文件 的 类 型 进行 控制 呢 ? | 
专家 解答 

限制 上 传 文件 类 型 ， 首 先 要 通过 预定 义 全 局 变量 $_FILES 中 的 $_ FILES[files'][mame'] 获 
取 上 传 文件 的 名 称 ， 然 后 应 用 strstr0 截 取 上 传 文件 的 后 绥 ， 最 后 判断 上 传 文件 的 后 绷 是 否 符合 | 
要 求 。 | 

strstr(0) 函 数 的 语法 如 下 : 

string strstr(string strings, string needle); | 

该 函数 用 于 在 字符 串 strings 中 查找 字符 串 needle 的 位 置 ， 并 返回 从 needle 字符 串 开 始 到 | 
strings 字符 串 结束 所 有 的 内 容 ， 如 果 在 strings 中 没有 查找 到 needle， 则 该 函数 将 返回 false。 

判断 上 传 文件 类 型 的 方法 如 下 : 


<2php 
if(!empty($_FILES['up_picture'][ ‘hame']){ /判断 上 传 内 容 是 否 为 空 
$type=$_FILES[mp_picture][mame']; /获取 上 传 文件 的 名 称 
$types=strstr($type,.): /截取 上 传 文件 的 后 绷 
if($types—".txt"){ /判断 上 传 文件 的 后 绥 是 否 符合 要 求 
/省 略 了 部 分 代码 
} 
}else{ 
echo "上 传 文件 ".$_FILES[mp_pictute'][mame']." 类 型 不 正确 1 "; 
} 
} 
?> 
专家 点 评 


判断 上 传 文件 是 否 符合 多 种 类 型 中 的 一 种 最 佳 的 方法 就 是 将 多 种 类 型 定义 到 一 个 数组 中 ， | 
然后 通过 for 循环 输出 数组 中 的 元 素 值 ， 将 输出 的 元 素 值 与 获取 的 文件 后 级 进行 比较 ， 进而 判 | 
断 其 是 否 符合 要 求 。 | 


问题 137 ”如 何 通过 header0 函 数 进行 下 载 ? 


问题 讲述 


除了 可 以 通过 链接 方式 下 载 文件 之 外 ,还 可 以 通过 header 函数 进行 下 载 ， 于 么 这 个 方法 具 | 
体 是 如 何 实现 的 呢 ? | 


Ee UD 
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专家 解答 
通过 HITP 方式 下 载 文件 ， 主 要 应 用 header0 函 数 。 
1，headerO 函 数 


header() 函 数 属于 HTTP 函数 。 其 作用 是 以 HITP 协议 将 HTML 文档 的 标 头 送 到 浏览 器 ， 
并 告诉 浏览 器 具体 如 何 处 理 这 个 页 面 。headerO 函 数 的 语法 如 下 : 
Void header ( string string [, bool replace [, int http response _code]] ) 
参数 说 明 : 
@ string: 发 送 的 标 头 。 
@ replace: 如 果 一 次 发 送 多 个 标 头 ， 对 于 相似 的 标 头 是 替换 还 是 添加 。 如 果 是 false， 则 强 
制 发 送 多 个 同类 型 的 标 头 。 默 认 是 tue， 即 替换 。 
@ http_response_code: 强制 HITP 响应 为 指定 值 。 
通过 HTTP 下 载 的 代码 如 下 : 
header("Content-type: application/x-gzip"); 
header("Content-Disposition: attachment: filename= 文 件 名 "); 
header("Content-Description: PHP3 Generated Data"); > 
HTTP 标 头 有 很 多 ， 这 里 介绍 的 是 下 载 的 HITP 标 头 。 其 代码 如 下 : 
header('Content-Disposition: attachment filename="filename"); 
在 应 用 的 过 程 中 ， 唯 一 需要 改动 的 就 是 filename， 即 将 filename 替换 为 要 下 载 的 文件 。 
2. 应 用 示例 
根据 超 链接 传递 的 参数 值 判断 指定 的 文件 是 否 存 在 ， 如 果 存 在 ， 则 通过 headerO 函 数 完成 
文件 的 下 载 操作 。 其 关键 代码 如 下 : 


<?php 
header("Content-type: text/html: charset=UTF-8"): /设置 文件 编码 格式 
$path =$_GET[path]; // 获 取 文 件 路 径 
if(!empty($path) and !is_null(Spath)){ // 判 断 变 量 是 否 为 空 以 及 是 否 为 NULL 
S$filename=basename($path): 1/ 获取 文件 名 
Sfile=@fopen($path,"r"); 
if($file){ 
header("Content-type:application/octet-stream"): // 输 出 MIME 类 型 
header("Accept-ranges:bytes"): /接受 的 范围 单位 
header("Accept-length:".filesize($path)): /文件 长 度 
header("Content-Disposition:attachment:filename=".$filename); 
// 默 认 时 文件 保存 对 话 框 中 的 文件 名 称 
echo fread($file,filesize($path)): // 读 取 文 件 
felose($file); /关闭 文件 
exit: // 退 出 
}else{ 
echo "<script>alert(' 您 下 载 的 文件 不 存在 ! ); history.backQ;</script>"; 
} 
有 
?> 
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专家 点 评 
通过 header0 函 数 不 但 可 以 实现 文件 的 下 载 ， 还 可 以 实现 下 面 3 个 功能 : 
(1) 重 定向 ， 这 是 最 常用 的 功能 。 
header("Location: http://www.mrbecd.com"); 
(2) 强制 客户 端 每 次 访问 页 面 时 获取 最 新 资料 ， 而 不 是 使 用 存在 于 客户 端的 缓存 。 


/设置 页 面 的 过 期 时 间 ( 用 格林 威 治 时 间 表 示 )。 

header("Expires: Mon, 08 Ju 2008 08:08:08 GMT"); 

// 设 置 页 面 的 最 后 更 新 日 期 (用 格林 威 治 时 间 表 示 )， 使 浏览 器 获取 最 新 资料 
header("Last-Modified: " . gmdate("D, d MY H:i:s") . "GMT"); 


header("Cache-Control: no-cache， must-revalidate"); // 控 制 页 面 不 使 用 缓存 
header("Pragma: no-cache"); /参数 〈 与 以 前 的 服务 器 兼容 )， 即 兼容 HTTP1.0 协议 
header("Content-type: © application/file"); // 输 出 MIME 类 型 
header("Content-Length: & © 227685"); // 文 件 长 度 
header("Accept-Ranges: bytes"): // 接 受 的 范围 单位 

/默认 时 文件 保存 对 话 框 中 的 文件 名 称 


header("Content 一 Disposition: attachment; filename=$filename"); /实现 下 载 
(3) 输出 状态 值 到 浏览 器 ， 控 制 访问 权限 。 


header(HTTP/1.1 401 Unauthorized"):; 
header('status: 401 Unauthorized"):; 


问题 158 ”如 何 实现 多 文件 同时 上 传 ? 


问题 阐述 


上 传 图 片 到 服务 器 是 程序 开发 过 程 中 必 不 可 少 的 一 项 功能 。 使 用 该 功能 不 但 可 以 达到 图 片 | 
共享 的 效果 ， 而 且 可 以 提高 网 站 的 访问 量 ， 丰 富 网 站 的 内 容 。 但 是 ,在 网 站 的 后 台 管 理 系统 中 ， | 
需要 的 是 更 简单 、 更 快速 地 上 传 图 片 到 服务 器 ， 这 就 是 多 文件 同时 上 传 到 服务 器 。 那 么 这 个 功 | 
能 是 如 何 实现 的 呢 ? 它 的 原理 又 是 怎样 的 呢 ? | 


专家 解答 


多 文件 同时 上 传 的 关键 是 for 循环 语句 的 应 用 ， 通 过 for 循环 语句 定义 上 传 文件 的 文件 域 
名 称 ， 由 for 循环 语句 决定 一 次 最 多 可 上 传 几 个 文件 。 | 
在 处 理 提交 的 上 传 时 , 同样 应 用 for 循环 语句 , 获取 上 传 文件 的 数据 , 通过 move uploaded | 
file0 函 数 完成 文件 的 上 传 操作 。 | 
1. 定义 多 文件 上 传 表单 
<form action="" method="post" enctype="multipart/form-data"> 
<2php 
for($a=0:$a<4:$a++){ 
?> 
<input type="file" name="filename <?php echo $a;?>"> 
<2php 
} | 
本 UD 
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安 
<br><input class="one"type="submit"name="sub" value=" 上 传 "/> 
</form> 
> | 应 用 POST 方法 设置 enctype 的 属性 为 "multipart/form-data"， 将 数据 提交 到 本 页 ， 应 用 for 
多 一] if 语句 定义 文件 上 传 的 文件 域 。 


2，for 循环 语句 多 文件 上 传 处 理 


<2php 
ifisset($_ POST[sub'])){ 
for($a=0;$a<4;$att){ /循环 语句 
$name=$_ FILES['filename '.$al; // 将 变量 的 名 称 保存 在 变量 中 
Srand=rand(1,100000); // 获 取 随 机 数 
$name_type=substr($name["name"],-4,4); /获取 文件 名 称 的 后 4 位 即 文件 的 格式 
$time=microtime(): // 获 取 随 机 数 
$path="upload/.(($rand+ $time).$name type); /定义 上 传 文件 的 路 径 
$dir-upload/: /定义 目录 
这 move_uploaded file($name['tmp_name'],$path)){ 
echo "成 功 上 传 文件 ".($rand+$time).$name_type."<br>"; /输出 提示 
} // 移 动 到 服务 器 指定 位 置 
) 
} 
ye 


| 应 用 $_FILES 全 局 数组 以 for 循 环 语句 为 辅助 , 获取 上 传 数据 的 信息 ,通过 rand0 和 microtimeO 
| 函数 获取 的 数字 作为 文件 名 称 , 应 用 substr0 函 数 获取 上 传 文件 后 级 , 通过 move_uploaded file0 
| 函数 执行 文件 上 传 操作 。 

| 专家 点 评 

这 里 讲解 的 多 文件 上 传 的 方法 ， 其 文件 上 传 的 最 大 值 取决 于 for 循环 语句 设置 的 最 大 值 ， 

| 它 一 次 可 以 上 传 最 大 值 之 内 任意 多 个 文件 。 


问题 159 ”如 何 判 断 文 件 的 权限 ? 


| 问题 阐述 
文件 权限 的 判断 是 一 个 非常 重要 的 功能 ， 如 果 文 件 没有 读 、 写 的 权限 ， 那 么 就 不 能 对 其 进 
| 行 操作 ， 那 么 如 何 判断 文件 的 权限 呢 ? 
专家 解答 
| 判断 文件 是 否 具备 读 权限 应 用 的 是 is_readable0 函 数 ;， 判断 文件 是 否 具备 写 权 限 应 用 的 是 
，is_writable() 函 数 。 

is_readable() 函 数 ， 判 断 指 定 文件 是 否 可 读 ， 语 法 如 下 : 

bool is_readable ( string filename ) 
如 果 文 件 存在 并 且 可 读 则 返回 TRUE。 
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is_writable0 函数， 判断 指定 文件 是 否 可 写 ， 语 法 如 下 : | 
bool is_writable ( string filename ) | 
如 果 文 件 存在 并 且 可 写 则 返回 TRUE。filename 参数 可 以 是 一 个 允许 进行 是 否 可 写 检 查 的 | 


目录 名 。 | 鳃 
说 明 : 

PHP 也 许 只 能 以 运行 webserver 的 用 户 名 (通常 为 mobody' ) 来 访问 文件 。 不 计 入 安全 模式 
的 限制 。 | 

对 于 文件 权限 的 判断 是 对 文件 进行 操作 的 前 提 ， 特 别 是 在 执行 文件 的 读 取 、 写 入 、 导 
等 操作 时 ， 如 果 文 件 不 具备 读 、 写 的 权限 ， 那 么 这 些 操作 都 是 没有 任何 意义 的 。 
专家 点 评 | 

通过 is_readable0 和 is_writable0 函 数 可 以 判断 文件 是 否 具 有 读 、 写 的 权限 。 在 文件 系统 函 
数 库 中 还 可 以 通过 filepermsO 函 数 取 得 文件 的 权限 ， 它 的 返回 值 是 一 个 int 型 的 字符 串 。 | 


旺 


命名 | 


问题 160 ”如 何 获取 磁盘 分 区 的 大 小 ? 


问题 阐述 | 

通过 文件 系统 函数 不 但 可 以 对 目录 、 文件 进行 操作 , 而 且 可 以 获取 目录 、 文件 的 相关 信息 ， | 
那么 是 否 可 以 获取 磁盘 分 区 的 大 小 呢 ? | 
专家 解答 


获取 磁盘 分 区 的 大 小 应 用 的 是 disk total space0 函 数 ， 获 取 磁 盘 分 区 的 剩余 空间 应 用 的 是 
disk free_space0 函 数 。 | 
disk_total_space() 函 数 : 获取 一 个 目录 的 磁盘 总 大 小 ， 语 法 如 下 : 


float disk total space ( string directory ) | 
该 函数 根据 参数 directory 提供 的 一 个 目录 字符 串 , 返回 相应 的 文件 系统 或 磁盘 分 区 的 所 有 | 
字 节 数 。 | 
注意 : 
本 函数 不 能 作用 于 远程 文件 ， 其 检测 的 文件 必须 通过 服务 器 的 文件 系统 访问 。 
disk_free_space() 函 数 : 获取 一 个 目录 的 可 用 空间 ， 语 法 如 下 : 
float disk free_space(string directory) 
参数 directory 用 来 指定 文件 系统 或 磁盘 分 区 。 


说 明 : | 
本 函数 返回 的 是 该 目录 所 在 的 磁盘 分 区 的 总 大 小 ,因此 在 给 出 同一 个 磁盘 分 区 的 不 同 目录 | 
作为 参数 所 得 到 的 结果 完全 相同 。 | 


“215s 


人 ip 六 预 知道 的 300 个 问题 


pe 
&% 
应 用 示例 
本 示例 应 用 disk total space0) 函 数 获取 磁盘 分 区 大 小 。 其 具体 代码 如 下 : 
(1) 创建 index.php 文件 。 添 加 表单 ， 设 置 文 本 框 、 提 交 按 钮 ， 使 用 POST 方法 ， 将 数据 
| 提交 到 本 页 。 
| (2) 通过 $ POST 获取 表单 提交 的 目录 路 径 ， 首 先 ， 判 断 获取 的 目录 是 否 合理 。 然 后 ， 通 
过 iconv0 函 数 对 获取 的 字符 串 进行 编码 转换 。 接 着 ， 对 获取 的 字符 串 进 行 截取 ， 获 取 该 目录 所 
| 在 的 磁盘 分 区 ， 并 应 用 disk total space() 函 数 获取 磁盘 分 区 的 大 小 。 最 后 ， 应 用 opendir0 函 数 
| 和 readdir0 函 数 读 取 提 交 目 录 下 的 内 容 。index.php 的 关键 代码 如 下 : 
| <2php 
| if($_ POST [file name'] !="" && is file ($ POST [file name'] ) 一 false) { 
| $file name = iconv ("utf-8", "gb2312", $ POST [file name'] ); /编码 格式 转换 


$filesize_s=number format($filesize_s/(1024*1024*1024),2,".",""); // 字 节 的 格式 化 

echo "本 地 磁盘 (" .$dir . ") ”总 大 小 : " . $filesize z ." GB &nbsp; 可 用 空间 : 
" .$filesize_s." GB <br><br>"; 

echo $_POST ['file name'] . ”目录 下 的 内 容 :". "<br>"; 


if (file exists ( $file name )) { // 判 断 目录 是 否 存在 
Slen = strripos ( $file name, ":" ); // 截 取 字 符 串 
| $dir = substr ( $file name, 0, $len + 1 ); /获取 提交 目录 所 在 磁盘 
| Sfilesize z=disk total space ( $dir ): 1/ 获取 目录 总 大 小 
| $filesize_ z=number format($filesize Zz/(1024*1024*1024),2,".",""); /数字 的 格式 化 
| Sfilesize_s=disk free space ( $dir ): // 获 取 磁 盘 剩 余 空间 


$i=0; 
S$list = opendir ( $file_ name ); // 打 开 目 录 
while ( $read list = readdir ( $list ) ) { // 读 取 目 录 
| $i++; 
| echo "$i:". iconv ("gb2312", "utf-8", $read list )." <br> "; // 输 出 目录 中 的 内 容 
| } 
| closedir ( $list ); /关闭 目录 
| } else { 
| echo "<script>alert(' 目录 不 存在 ! "):</script>"; 
| b 
| } else { 
| echo "<script>alert( 请 输入 正确 的 目录 路 径 ! ");</script>"; 
} 
2 


运行 结果 如 图 7.2 所 示 。 
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专家 点 评 

strripos() 函 数 用 于 判断 指定 字符 串 在 另 一 字符 串 中 最 后 出 现 的 位 置 。 通 过 本 函数 可 以 获取 
到 指定 字符 串 A 在 另 一 字符 串 B 中 最 后 出 现 的 位 置 ， 其 返回 值 为 int 型 ， 根 据 该 返回 值 就 可 以 | 
对 字符 串 B 以 字符 串 A 为 分 隔 点 进行 截取 ， 在 执行 截取 的 操作 时 需要 应 用 到 substr() 函 数 。 | 


S$len = strripos ( $file name, ":" ); // 截 取 字符 串 
$dir = substr ( $file name, 0, $len + 1): /获取 提交 目录 所 在 磁盘 


问题 161 如 何 遍 历 指 定 目 录 下 的 所 有 文件 ? 


问题 阐述 | 
在 网 站 的 后 台 管理 系统 中 ,经 常 需要 对 网 站 服务 器 中 的 文件 进行 管理 和 维护 ， 有 时 需要 添 
加 一 个 文件 来、 删除 某 个 文件 夹 或 文件 ， 为 了 可 以 更 好 地 查看 到 这 些 文件 或 文件 夹 ， 就 需要 对 | 
服务 器 指定 目录 下 的 文件 进行 遍历 。 如 何 实现 这 个 指定 目录 下 文件 的 遍历 功能 呢 ? | 
专家 解答 
遍历 指定 目录 下 的 所 有 文件 的 关键 是 目录 和 文件 处 理 函数 的 应 用 ， 包 括 file_exists()、 
is_dir0)、opendir0、readdir0 和 closedir0 〇 函数 。 
1. 应 用 函数 解析 
(1) file_exists(): 检查 文件 或 目录 是 否 存 在 。 语 法 如 下 : 
bool file_exists(string filename) 
如 果 由 flename 指定 的 文件 或 目录 存在 ， 则 返回 rue， 否则 返回 false。 
(2) is_dir0: 判断 给 定 的 文件 名 是 否 是 一 个 目录 。 
bool is_dir(string filename) 
如 果 文 件 名 存在 并 且 为 目录 ， 则 返回 tue。 如 果 fename 是 一 个 相对 路 径 ， 则 按照 当前 工 
作 目 录 检 查 其 相对 路 径 。 
说 明 : | 
本 函数 的 结果 会 被 缓存 ， 且 其 不 能 作用 于 远程 文件 ， 被 检查 的 文件 必须 通过 服务 器 的 文件 | 
系统 访问 。 | 
(3) opendir0: 打开 目录 句柄 。 语 法 如 下 : 
Tesource opendir ( string path) | 
本 函数 返回 一 个 目录 句柄 ， 可 以 在 dlosedir0、readdir0 之 前 调用 。 如 果 pa 不 是 一 个 合法 的 目 | 
录 或 者 因为 权限 限制 或 文件 系统 错误 而 不 能 打开 上 有 目录，opendir0 返 回 false 并 产生 一 个 错误 信息 。 | 
(4) readdir0: 从 目录 句柄 中 读 取 条 目 。 语 法 如 下 : | 
string readdir( resource dir handle) 
本 函数 返回 目录 中 下 一 个 文件 的 文件 名 。 文 件 名 以 在 文件 系统 中 的 排序 返 
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(5) closedir0: 关闭 目录 句柄 。 语 法 如 下 : 
string closedir( resource dir handle) 
g | 本 函数 关闭 由 dir_handle 指定 的 目录 流 ， 目 录 流 必须 被 opendir0 打 开 过 。 
全 ， ”综合 运用 上 述 函 数 ， 就 可 以 实现 指定 目录 下 的 文件 遍历 操作 。 
2. 应 用 示例 
| 在 本 示例 中 应 用 上 述 介绍 的 函数 ， 实 现 对 指定 目录 下 文件 的 遍历 操作 。 其 具体 步骤 如 下 : 

(1) 创建 index.php 文件 。 添加 一 个 表单 , 定义 表单 元 素 , 通过 post 方法 提交 目录 的 路 径 。 
| (2) 在 indexphp 中 ， 判 断 提交 的 目录 路 径 是 否 为 空 ， 如 果 不 为 空 ， 则 打开 目录 ， 读 取 目 
| 录 中 的 文件 ， 并 输出 读 取 到 文件 ， 关 闭 目 录 ; 否则 ， 输 出 目录 不 存在 。 其 关键 代码 如 下 : 


<?php 
这 !file_exists ( $look file )){ /检测 指定 的 目录 是 否 存在 
print $look file "目录 不 存在 " : 
}else{ 
$i=0; 
if(is_dir ( $look file )){ /检测 是 否 是 合法 目录 
if ($list = opendir ( $look file )){ // 打 开 目 录 
while ($read list = readdir( $list )){ // 读 取 目 录 
S$i++ ; 
echo " $i:".iconv("gb2312","utf-8",$read_list)." <br>"; ”// 输 出 目录 中 的 内 容 
} 
b 
} 
closedir ( $list ) ; // 关 闭 目 录 
b 
2 


| 运行 本 示例 ， 在 文本 框 中 输入 一 个 指定 的 文件 夹 ， 单 击 “ 提 交 ” 按 钮 ， 如 果 该 文件 夹 存在 ， 
| 就 可 以 显示 出 该 文件 夹 下 包括 的 所 有 文件 。 运 行 结 果 如 图 7.3 所 示 。 


A 请 的 个 人 博客 
WD -三 办 
目录 位 置 : 二 | 
二 
3: inages 
4: index.php 


Copytight © 1999-2010 吉林 各 明日 和 技 有 限 公司 


图 7.3 遍历 指定 目录 下 的 所 有 文件 
问题 162 ”如 何 应 用 fopen() 函 数 的 选择 模式 ? 


问题 阐述 
如 何 应 用 fopen0) 函 数 的 选择 模式 ? 
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1. fopen 函数 
在 PHP 中 使 用 fopen0 函 数 打开 一 个 文件 ， 语 法 如 下 : 
Tesource fopen (string filename, string mode [, int use_include_path [, resource zcontext]]) 
参数 flename 指定 打开 的 文件 名 。 
注意 : | 
参数 filename 可 以 是 包含 文件 路 径 的 文件 名 ( 例如 “C:/windows/php.ini” 或 “./php.ini” )。 | 
为 了 避免 不 同系 统 之 间 切 换 可 能 带 来 的 麻烦 ， 采 用 “/” 作 为 路 径 分 隔 符 。 参数 filename 也 可 以 | 
是 由 某 种 协议 给 出 的 URL (例如 “http://mrbccd.cn” 或 “ftp://wwwmrbced.cn//”)。 如 果 指 定 | 
URL 地 址 ， 则 可 以 打开 远程 文件 。 | 
参数 mode 设置 打开 文件 的 方式 ， 参 数值 如 表 72 所 示 。 
表 7.2 fopen() 中 参数 mode 的 可 选 值 | 


mode [模式 名 称 说 了 明 | 
r 只 读 读 模式 一 一 读 文件 ， 文 件 指针 位 于 文件 头 部 | 
和 读 写 模式 一 一 读 、 写 文件 ， 文 件 指针 位 于 文件 头 部 。 注 意 ， 如 果 在 现 有 文件 内 容 
的 未 尾 之 前 进行 写 入 就 会 缆 盖 原 有 内 容 
下 写 模式 一 一 写 入 文件 ， 文 件 指针 指向 文件 头 部 。 注 意 ， 如 果 文件 存在 ， 则 文件 内 
起 容 被 删除 ， 重 新 写 入 ， 如 果 文 件 不 在 在 ， 则 函数 自行 创建 文件 
a 珊 重 写 模式 一 一 读 、 写 文件 ， 文 件 指针 指向 文件 头 部 。 注 意 : 如 果 文件 存在 ， 则 文件 


内 容 被 删除 ， 重 新 写 入 ; 如 果 文 件 不 存在 ， 则 函数 自行 创建 文件 
写 模式 打开 文件 ， 从 文件 头 部 开始 写 入 。 注 意 : 如 果 文 件 已 经 存在 ， 函 数 返 
FALSE， 产 生 一 个 E_ WARNING 级 别 的 错误 

读 / 写 模式 打开 文件 ， 从 文件 头 部 开始 写 入 。 注 意 
FALSE， 产 生 一 个 E_ WARNING 级 别 的 错 训 
追加 模式 打开 文件 ， 文 件 指针 指向 文件 尾部 。 注 意 : 如 果 文 件 已 有 内 容 ， 则 将 从 
文件 末尾 开始 追加 ; 如果 文 件 不 存在 ， 则 函数 自行 创建 文件 

追加 模式 打开 文件 ， 文 件 指针 指向 文件 尾部 。 注 意 : 如 果 文 件 已 有 内 容 ， 则 从 文 | 
件 末尾 开始 追加 或 读 取 ;如果 文件 不 存在 ， 则 函数 自行 创建 文件 | 
二 进 制 模式 一 一 用 于 与 其 他 模式 进行 连接 。 注 意 : 如 果 文件 系统 能 够 区 分 二 进 制 | 
b 二 进 制 ”| 文件 和 文本 文件 ， 可 能 会 使 用 它 。Windows 可 以 区 分 ; UNIX 则 不 区 分 。 推 荐 使 用 | 
这 个 选项 ， 便 于 获得 最 大 程度 的 可 移植 性 。 它 是 默认 模式 
上 文本 用 于 与 其 他 模式 的 结合 。 该 模式 只 是 Windows 下 的 一 个 选项 。 | 


瑟 


ne 如 果 文 件 已 经 存在 ,函数 返回 
x+ 谨慎 写 


at 追加 


参数 use_include_path 为 可 选 参数 ， 决 定 是 否 在 include_path〈php.ini 中 的 include path 选 | 
项 ) 定义 的 目录 中 搜索 filename 文件 。 例如， 在 php.ini 文件 中 设置 include_path 选项 的 值 为 ， 
“EE:\AppServiwww\MR\Instance\07\”， 如 果 希 望 服务 器 在 这 个 路 径 下 打开 所 指定 的 文件 ， 则 设 
置 参 数 use_include_path 的 值 为 1 或 TRUE。 

参数 context 称 为 上 下 文 ， 同 样 为 可 选 参数 ， 它 是 设置 流 操作 的 特定 选项 ， 用 于 控制 流 的 
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操作 特性 。 一 般 情况 下 只 需 使 用 默认 的 流 操作 设置 ， 不 需要 使 用 此 参数 。 

2. 模式 选择 

不 同 的 操作 系统 具有 不 同 的 行 结束 习惯 。 当 要 在 文本 文件 中 插入 一 个 新 行 时 ， 对 于 行 结束 
符 的 选择 必须 适合 用 户 的 操作 系统 ， 如 果 使 用 错误 的 行 结束 符 ， 那 么 当 打 开 文 件 时 ， 输 出 的 可 
能 是 一 堆 乱 码 。 所 以 正确 地 使 用 行 结束 符 是 非常 重要 的 。 

基于 UNIX 的 系统 使 用 \n 作为 行 结束 符 ， 基 于 Windows 的 系统 使 用 rm 作为 行 结束 符 ， 基 


| 于 Macintosh 的 系统 使 用 \r 作为 行 结束 符 。 


在 Windows 下 提供 一 个 文本 转换 标记 () 可 以 将 n 转换 为 rn; 或 者 也 可 以 使 用 ('b') 设 


| 置 二 进 制 模式 ,这样 就 不 会 对 数据 进行 转换 。 所 以 ,在 Windows 系统 要 使 用 行 结束 符 有 两 个 选 
| 择 : 一 使 用 文本 转换 标记 〈Y) 进行 转换 ;二 使 用 ('b') 设置 mode 参数 的 模式 为 二 进 制 。 


默认 的 转换 模式 依赖 于 SAPI 和 使 用 的 PHP 版 本 。 因 此 为 了 便于 移植 ， 建 议 在 应 用 fopen() 


| 函数 时 要 指定 恰当 的 模式 。 


如 果 想 让 以 \n 作为 行 结束 符 的 纯 文本 文件 在 其 他 应 用 程序 (如 Notepad) 中 也 有 效 ， 那 么 


建议 在 在 脚本 中 使 用 〈Y) 模式 ， 在 其 他 情况 下 使 用 ('b')。 


如 果 在 操作 三 进 制 文 件 时 没有 指定 ('b') 模式 ， 可 能 会 输出 一 些 奇怪 的 内 容 。 因 此 ， 考 虑 


到 程序 的 可 移植 性 ， 强 烈 建议 在 用 fopen0 打 开 文 件 时 总 是 使 用 〈"b') 模式 ， 同 时， 建议 重 写 那 
| 些 依赖 于 〈t) 模式 的 代码 ， 使 其 使 用 正确 的 行 结束 符 和 正确 的 ('b') 模式 。 


问题 163 ”如 何 镇 定 文件 ? 


| 问题 阅 述 


| 在 向 一 个 文本 文件 写 入 数据 时 ， 需 要 先 锁定 文件 ， 以 防 其 他 用 户 同时 也 修改 此 文件 内 容 。 
| 如 何 锁定 文件 ? 


| 专家 解答 


PHP 中 通过 flockO 函 数 实现 文件 锁定 ， 语 法 如 下 : 
bool flock (int handle, int operation) 
参数 handle 是 一 个 已 经 打开 的 文件 指针 ; 参数 operation 控制 锁定 权限 ， 有 4 个 可 选 值 : 
LOCK_SH: 取得 共享 锁定 〈 读 取 程 序 )。 
LOCK EX: 取得 独占 锁定 〈 写 入 程序 )。 
LOCK _ UN: 释放 锁定 。 
LOCK NB: 防止 flock0 在 锁定 时 堵塞 。 


问题 164 ”如 何 访问 远程 文件 ? 


”问题 阐述 


如 何 访问 远程 文件 ? 
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在 PHP 中 访问 远程 文件 使 用 fopen0 函 数 ， 前 提 是 将 配置 文件 php.ini 中 的 选项 allow ur | 
fopen 设置 为 ON。allow_url fopen 参数 默认 是 开启 的 , 允许 打开 HTTP 协议 和 FTP 协议 指定 的 | 
远程 文件 , 如 果 allow_url fopen 设置 为 OFF, 则 不 允许 打开 远程 文件 , fopen0 函 数 将 返回 false。 | 对 J 

例如 ， 首 先 将 php.ini 中 的 选项 allow_url fopen 设置 为 OFF， 然 后 应 用 fopen0 函 数 访问 一 
个 远程 文件 ， 最 后 通过 readfile0 函 数 读 取 文件 内 容 。 代 码 如 下 : | 


<?php 
/访问 远程 文件 
$filename = "http://192.168.1.249/admin/file/20080330924147598412.txt"; /要 读 取 的 文件 
$f = fopen($filename,"rb"); /打开 文件 

Ye 


由 于 php.ini 中 的 选项 allow_url fopen 设置 为 OFF， 所 以 此 时 访问 远程 文件 失败 ， 将 返回 
如 图 7.4 所 示 的 错误 信息 。 | 


图 7.4 访问 远程 文件 失败 
问题 165 如何 读 取 本 地 文件 ? 


问题 阐述 

读 取 本 地 文件 的 方法 有 哪些 ? 
专家 解答 | 
读 取 文件 主要 是 3 种 功能 : 读 取 文件 到 数组 中 、 a | 
1.，file 函数 读 取 文 件 到 数组 中 


file0) 函 数 将 整个 文件 的 内 容 读 入 到 一 个 数组 中 。 成 功 返 回 数 组 ,数组 中 9 全 元素 才 是 | 
件 中 对 应 的 一 行 ， 包 括 换 行 符 在 内 ;失败 返回 FALSE。 语 法 如 下 : 
array file ( string filename [, int use_include_path [, resource context]] ) 
参数 flename 指定 读 取 的 文件 名 称 ; 参数 use_include_path 控制 是 否 支 持 在 include_path 中 | 
搜索 文件 ， 如 果 支 持 ， 则 将 该 值 设置 为 TRUE; 参数 context 是 PHP5.0 新 增 内 容 。 | 
PHP 的 file0 函 数 将 文件 读 取 到 数组 中 ， 在 读 取 过 程 中 使 用 新 行 符 指示 新 数组 元 素 从 哪里 | 
开始 。 其 具体 应 用 如 下 : | 


<2php 
$file=file('zhuce.php"); 
Slines=count($file); 
$alt="; 

for($i=0; $1<S$lines;$1++){ 
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$alt=($alt 一 even)2odd:even': 
echo'<div class=".$alt.™">"; 

echo $i.':'.htmlspecialchars($file[$1]); 
echo"</div>\n"; 


以 上 程序 将 文件 输出 到 一 个 格式 良好 的 页 面 中 ， 因 此 可 以 逐 行 地 进行 检查 。 使 用 for 循环 
读 取 $file 数组 变量 , 并 按照 希望 的 格式 显示 。 在 for 循环 语句 中 应 用 三 元 运算 符 设置 行 的 颜色 。 
2.，file_get_contents 函数 读 取 文 件 到 字符 串 中 
file_get_contents() 函 数 将 文件 内 容 读 入 一 个 字符 串 。 如 果 有 offset 和 maxlen 参数 ， 将 在 参 
数 offset 所 指定 的 位 置 开始 读 取 长 度 为 maxlen 的 内 容 。 如 果 失 败 ， 则 返回 FALSE。 语 法 如 下 : 
string file_get_contents ( string filename [, bool use_include_path [, resource context [, int offset [, int 
maxlen]]]] ) 
参数 flename 指定 读 取 的 文件 名 称 ; 参数 use_include_path 控制 是 否 支 持 在 include_path 中 
搜索 文件 ， 如 果 支 持 ， 则 将 该 值 设 置 为 TRUE。 
PHP 从 4.3 版 本 开始 包含 file_get_contents 的 函数 , 该 函数 能 够 在 不 分 解 文件 的 情况 下 直接 
将 文件 读 取 到 字符 串 中 。 其 具体 应 用 如 下 : 
<2php 
$context = array( 
‘http' => array( 
‘proxy’ => 'tep://127.0.0.1', 
Tequest_fulluri => true, 


) 


Geis = stream context_create($context); 
Sfiles = file_get_contents("http://www.mrbced.com", false, $contexts); 
echo $files; 
> 
3. readfile 函数 直接 读 取 文件 
readfileO 函 数 读 取 一 个 文件 并 写 入 输出 缓冲 ， 如 果 成 功 返回 则 读 取 字 节 数 ， 如 果 失 败 则 返 
回 FALSE。 语 法 如 下 : 
int readfile ( string filename [, bool use_include_path [, resource context]] ) 
参数 filename 指定 读 取 的 文件 名 称 ; 参数 use_include_path 控制 是 否 支持 在 include_path 中 
搜索 文件 ， 如 果 支 持 ， 则 将 该 值 设置 为 TRUE; 参数 context 是 PHP 5.0 的 新 增 内 容 。 


注意 : 
应 用 readfile 〇 0 函数， 不 需要 打开 /关闭 文件 ， 不 需要 输出 语句 ， 直 接应 用 函数 即 可 。 


专家 点 评 


Readfile 是 一 种 保护 文件 和 带宽 的 便捷 方法 。 通 过 一 个 使 用 readfile 函数 的 脚本 将 用 户 Web 
站 点 中 的 所 有 文件 链接 起 来 , 能 够 防止 其 他 站 点 直接 链接 到 它们 , 从 而 阻止 其 对 Web 站 点 带宽 
的 可 能 占用 。 这 种 方法 使 用 的 技术 通常 被 称 为 “ 反 盗 链 ”。 如 果 引 入 了 验证 系统 和 /或 HTTP 引 
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用 检查 ， 就 拥有 了 一 个 确保 只 有 合法 参观 者 能 够 访问 您 文件 的 安全 系统 。 


问题 166 ”如 何 使 用 文件 句柄 ? 


问题 阐述 


为 了 使 用 读 取 文件 函数 ， 需 要 使 用 相对 于 PHP 脚本 的 路 径 为 它们 指出 要 读 取 的 文件 。 然 | 
而 ， 大 多 数 PHP 的 文件 函数 使 用 一 种 稍 有 区 别 的 机 制 访问 文件 ， 这 种 机 制 与 连接 数据 库 的 机 | 
制 非常 相似 。 该 过 程 使 用 fopen 函数 进行 “连接 ”， 使 用 fclose 函数 “ 断 开 连接 ”。fopen 函数 的 | 
返回 值 是 一 个 PHP 文件 指针 ， 也 被 称 为 文件 句柄 。 一 旦 拥有 了 文件 句柄 ， 就 可 以 使 用 它 对 文 | 
件 执 行 多 种 操作 , 包括 读 取 、 附加 操作 等 。 那 么 这 个 句柄 在 使 用 过 程 中 都 需要 注意 哪些 问题 呢 ? 


专家 解答 
首先 来 看 一 下 文件 是 如 何 打开 和 关闭 的 。 其 代码 如 下 : 
<2php 
$f open =fopen("../file.txt.","rb"); /打开 文件 
站 // 对 文件 进行 操作 
fclose($f open) /操作 完成 后 关闭 文件 
PS 


使 用 fopen 连接 文件 时 ， 必 须 指定 文件 路 径 和 文件 访问 模式 (例如 ，r 表示 只 读 模式 )。b 
模式 表示 文件 将 以 二 进 制 模式 打开 ， 二进制 模式 可 以 用 来 保证 代码 在 不 同 操作 系统 之 间 具 体 可 | 
1. 通过 句柄 定位 文件 读 取 的 起 始 位 置 | 
在 通过 fopen0 函 数 打 开 一 个 文件 值 时 就 已 经 拥有 了 一 个 文件 句柄 , 下 面 就 应 用 该 句柄 读 取 | 
文件 。 其 代码 如 下 : | 


<2php 
Sfilename = "04.txt"; // 要 读 取 的 文件 
$fp = fopen($filename,"rb"); /打开 文件 
echo fread($fp,32); /使 用 fread0 函 数 读 取 文 件 内 容 的 前 32 个 字 节 
Ee 
echo fread($fp,filesize($filename)): // 输 出 其 余 的 文件 内 容 
?> 


这 个 例子 演示 了 如 何 使 用 文件 句柄 。 注 意 ， 当 使 用 fread 时 ， 第 二 个 参数 以 字 节 为 单位 ， 
表示 从 文件 开始 读 取 的 数据 量 。 文 件 内 容 被 读 取 到 设置 的 数据 量 时 ， 句 柄 将 定位 到 这 个 位 置 ， | 
如 果 想 要 继续 读 取 这 个 文件 中 的 其 他 内 容 ， 则 可 以 通过 filesize 函数 获取 文件 的 总 大 小 ， 并且 | 
将 这 个 值 作为 fread 的 第 二 个 参数 ， 此 时 输出 的 将 是 文件 剩余 部 分 的 内 容 。 这 就 是 句柄 的 作用 。 | 

2， 大 文件 的 分 块 读 取 | 
前 面 的 方案 对 于 小 文件 读 取 非 常 有 效 ， 但 是 在 读 取 较 大 文件 的 内 容 时 ， 服 务 器 的 内 存 将 被 | 
占用 ， 由 此 可 能 会 造成 性 能 问题 。 为 了 减少 发 生 该 问题 的 可 能 , 可 以 采用 以 块 读 取 文件 的 方法 ，| 
并 以 块 为 对 象 进行 操作 ， 这 样 就 避免 了 对 内 存 的 占用 。 其 原理 是 ， 通 过 fopen 函数 打开 文件 ， 


Re L_ 
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应 用 while 语句 循环 读 取 文 件 的 内 容 , 当 文 件 指针 到 达 文 件 结束 位 置 时 停止 循环 ; 否则 应 用 fgets 
函数 读 取 文件 中 数据 ， 并 且 向 前 移动 文件 句柄 。 以 块 读 取 文件 的 具体 代码 如 下 : 


<?php 
$fopen = fopen('fun.php','rb'); 
while(!feof($fopen)){ /使 用 feofO 函 数 测试 指针 是 否 到 了 文件 结束 的 位 置 
echo feets($fopen); // 输 出 当前 行 
} 
fclose($fopen); 

! Ts 
| 专家 点 评 


还 有 很 多 函数 能 够 使 用 文件 句柄 读 取 文 件 。 其 中 一 个 是 fgetss 函数 〈 注 意 是 两 个 s)， 它 除 


了 能 够 像 strip_tags 函数 那样 剥离 其 发 现 的 所 有 HTML 标记 之 外 ， 其 他 方面 几乎 与 fgets 相同 。 
| 另 一 个 是 fscanf 函数 ， 它 像 printf 一 样 格式 化 来 自 文件 的 输出 。 另 外 还 有 fgetcsv， 它 使 得 处 理 
| csv 〈comma separated values， 逗 号 分 割 值 ) 文件 变 得 十 分 容易 。 


但 是 如 果 希 望 将 文件 的 全 部 内 容 都 读 取 到 一 个 变量 中 ， 那 么 fille 和 file_get_ content 函数 更 


| 易于 使 用 ， 它 们 还 潜在 地 提供 了 更 好 的 性 能 。 


问题 167 如 何在 文件 中 存储 配置 信息 ? 


| 问题 阐述 


某 些 在 站 点 中 重复 使 用 的 信息 (如 密码 、 路 径 和 变量 ) 最 好 存储 在 一 个 文件 中 。 这 样 ， 如 


果 需 要 把 代码 转移 到 另 一 个 站 点 ， 就 只 需要 修改 一 次 设置 ， 而 不 用 成 百 上 千 次 地 修改 代码 。 那 
| 么 如 何 将 这 些 配置 信息 存储 到 指定 的 文件 中 的 呢 ? 又 如 何 读 取 这 些 配 置信 息 呢 ? 


专家 解答 


最 简单 的 存储 配置 信息 的 方法 是 在 一 个 .ini 文件 中 建立 变量 ， 然 后 使 用 parse_ini_file 函数 


在 代码 中 包含 这 个 文件 ， 该 函数 能 够 解析 与 php.ini 相同 格式 的 文件 。 例 如 ， 下 面 建 立 一 个 
| config ini 文件 。 其 内 容 如 下 : 


; This is a mysql configuration file 

; Comments start with ';', as in php.ini 
host = localhost 

dbname = db_database 

Username = root 


password = 111 
charset = utf8 
应 用 parse_ini_file 函数 读 取 配置 文件 config.ini 中 的 数据 ， 其 代码 如 下 : 
<?php 
S$ini array = parse_ini file("config.ini"); 
Print T($ini array); 


S$ini array = parse_ini file("config.ini", true); 
Print ($ini array); 
> 
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输出 结果 如 下 : 


Array 
( 


[host] => localhost 
[dbname] => db_database 
[usermame] => root 


[password] => 111 
[charset] => utf8 

) 

Array 

{ 
[host] => localhost 
[dbname] => db_database 
[usermmame] => root 
[password] => 111 
[charset] => utf8 

) 

专家 点 评 


在 存储 配置 文件 时 ,最 好 不 要 将 配置 文件 存储 于 Web 根 目录 下 , 因为 配置 文件 通常 会 包含 | 
用 户 名 和 密码 信息 。 如 果 将 配置 文件 存储 于 Web 根 目 录 下 ， 则 很 容易 被 “不 速 之 客 ” 访问 。 所 | 
以 ， 还 是 应 将 其 存储 于 服务 器 之 外 ， 这 样 “ 不 速 之 客 ” 就 很 难 访问 这 个 文件 了 。 | 

如 果 必 须 将 配置 文件 存储 到 Web 根 目 录 下 ， 一 定 要 在 .htaccess 文件 中 包含 一 条 文件 指令 ， 
用 于 限制 能 够 访问 该 类 文件 的 用 户 。 


问题 168 ”如 何在 PHP 中 创建 压缩 的 RAR 文件 ? 


问题 阐述 
在 网 站 中 ， 上 传 的 文件 大 多 数 都 是 以 压缩 包 的 形式 存在 的 ， 所 以 先 将 要 上 传 的 文件 压缩 后 
才能 够 上 传 ， 那 么 如 何 实现 网 站 中 上 传 文件 的 自动 压缩 功能 呢 ? | 
专家 解答 | 
文件 压缩 主要 应 用 PHP 的 预定 义 类 com, 将 wseriptshell 对 象 作为 com 类 的 参数 ,最 终 通 | 
过 调用 com 类 的 run0 方 法 实现 文件 的 压缩 。 | 
com 类 提供 一 个 将 COM 组 件 整 合 到 PHP 脚本 中 的 框架 。 实 例 化 方法 如 下 : 
$obj = new COM("server.object") 
com 类 构造 函数 的 语法 如 下 : 
string COM::COM ( string module_name [, string server_name [, int codepage]] ) 
参数 说 明 : 
@ module name: 被 请 求 组 件 的 名 字 或 class-id。 | 
@ server name: DCOM 服务 器 的 名 字 ， 组 件 在 此 服务 器 上 被 取 用 。 如 果 是 NULL， 则 假 | 
定 是 localhost。 如 果 要 支持 允许 DCOM, 则 必须 将 php.ini 中 的 com.allow_dcom 设置 为 TRUE。 | 
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pe 
可 

日 codepage: 指定 用 于 将 PHP 字符 串 〈php-strings) 转换 成 UNICODE 字符 串 (unicode- 
strings) 的 代码 页 , 反之 亦 然 。 可 用 的 值 有 CP_ACP、CP_MACCP、 CP_OEMCP、 CP_SYMBOL、 
CP_THREAD ACP, CP UTF7 和 CP UTF8.。 

com 类 中 run() 方 法 的 语法 如 下 : 

run("winrar a 生成 的 rar 文件 的 路 径 及 名 称 要 压缩 的 文件 的 路 径 及 名 称 ",1Lfalse); 

应 用 示例 

本 示例 中 讲解 在 PHP 中 如 何 创建 RAR 文件 。 其 具体 步骤 如 下 : 

(1) 创建 index.php 文件 。 首 先 ， 创 建 一 个 表单 ， 设 置 文件 域 ， 提 交 要 压缩 的 文件 ， 设 置 
文本 框 ， 提 交 压 缩 文件 的 存储 位 置 ， 设 置 按钮 ， 完 成 表单 元 素 的 提交 操作 。 

(2) 在 indexphp 文件 中 编写 PHP 脚本 ， 通 过 $_ POST[] 方 法 获取 表单 提交 的 数据 ， 通 过 
iconv0) 函 数 完成 数据 编码 格式 的 转换 ， 实 例 化 com 类 ， 调 用 run0 方 法 完成 文件 的 压缩 操作 。 
其 关键 代码 如 下 : 
<2php 


if($_POST ['Submit] (="") { 
$sourcefile = iconv ("utf-8", "gb2312", $_ POST ['sourcefile'] ); /获取 指定 文件 的 路 径 


S$obijfile = $_POST [objfile] .time 0; /定义 压缩 文件 的 存储 位 置 和 名 称 
$obifile = iconv ( "utf-8", "gb2312", $objfile ); /实现 编码 转换 
$obj = new com ( "wscript.shell" ): /实例 化 com 类 


$obj->run ( "winrara". $objfile ."". $sourcefile . "", 1, false ); V/ 调 用 类 中 的 方法 
echo "<scripf>alert( 文 件 讨 缩 完成 !):</script>": 
} 


?> 
运行 本 示例 ， 首 先 选择 要 压缩 的 文件 ， 然 后 设置 压缩 文件 的 存储 位 置 ， 单 击 “ 提 交 ” 按 钮 
后 。 其 运行 结果 如 图 7.5 所 示 。 


一 
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设 雪 后 瑞 文件 存 由 人 置 ( 的 如 : CF\AppSemvwwwWRO4025) 1 
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图 7.5 RAR 压缩 文件 
专家 点 评 


在 执行 文件 压缩 操作 时 , 切记 在 文件 的 存储 路 径 中 不 要 出 现 空格 (包括 被 压缩 文件 的 路 径 、 
名 称 以 及 压缩 文件 的 存储 路 径 )， 和 否则 压缩 操作 将 不 会 正确 执行 。 


问题 169 ”如 何在 PHP 中 对 RAR 文件 进行 解压 ? 


问题 曾 述 


在 PHP 中 既然 可 以 对 文件 进行 压缩 ， 那 么 就 一 定 可 以 对 这 个 压缩 的 文件 进行 解压 ， 这 个 
解压 的 功能 又 是 如 何 实现 的 呢 ? 
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专家 解答 


文件 解压 同样 应 用 PHP 的 预定 义 类 com， 将 wscript.shell 对 象 作 为 com 类 的 参数 ， 通 过 调 | 
用 com 类 的 mn0 方 法 执行 文件 解压 的 命令 。 文 件 解压 的 命令 如 下 : | 
$obj->run ("winrar x -av -c ".$sourcefile." ".$objfile. ", 1, true ); /调用 类 中 的 方法 , 执行 文件 解压 的 命令 | 
通过 此 命令 将 指定 的 压缩 文件 解压 到 指定 的 绝对 路 径 下 。 
应 用 示例 
本 示例 对 压缩 的 RAR 文件 进行 解压 。 其 具体 步骤 如 下 : | 
(1) 创建 index.php 文件 。 首 先 ， 创 建 一 个 表单 ， 设 置 文 件 域 ， 提 交 要 解压 的 文件 ， 设 置 | 
文本 框 ， 提 交 解 压 后 文件 的 存储 位 置 ， 设 置 按钮 ， 完 成 表单 元 素 的 提交 操作 。 | 
(2) 在 indexphp 文件 中 编写 PHP 脚本 ， 通 过 $_POST[] 方 法 获取 表单 提交 的 数据 ， 通 过 
iconv0 函 数 完成 数据 编码 格式 的 转换 ， 实 例 化 com 类 ， 调 用 run0 方 法 执行 文件 的 解压 命令 。 
其 关键 代码 如 下 : 


<?php 
if($_POST [Submit] (="") { 
include("function.php"); 
$sourcefile=iconv("utf-8", "gb2312", unhtml($_POST['sourcefile"])); /获取 指定 文件 的 路 径 
$objfile=iconv( "utf8", "gb2312", $ POST ['objfile] );// 定 义 解压 文件 的 存储 位 置 和 名 称 
$obj = new com( "wscript.shell" ); // 实 例 化 com 类 | 
$obj->run ("winrarx -av -c ".$sourcefile." ".$objfile. "", 1, true ); // 调 用 类 中 的 方法 , 执行 | 
解压 命令 | 
echo "<script>alert( 文 件 解 讨 完 成 !);</script>"; 
» 
> 


运行 本 示例 ， 首 先 选 择 要 解压 的 RAR 文件 ， 并 在 文本 框 中 输入 要 解压 文件 的 存储 目录 ， 
然后 单 击 “解压 ”按钮 即 可 。 其 运行 结果 如 图 7.6 所 示 。 


< | 
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图 7.6 解压 RAR 文件 


专家 点 评 
在 执行 文件 解压 操作 时 , 切记 在 文件 的 存储 路 径 中 不 要 出 现 空格 (包括 被 解压 文件 的 路 径 、 | 

名 称 以 及 解压 后 文件 的 存储 路 径 )， 和 否则 解压 操作 将 不 会 正确 执行 。 | 
问题 170 ”如 何 将 文本 文件 中 的 数据 存储 到 数据 库 中 ? 


问题 讲述 
如 何 将 文本 文件 中 的 数据 存储 到 数据 库 中 ? | 
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专家 解答 


| 将 文本 文件 中 的 数据 存储 到 数据 库 中 主要 运用 到 3 方面 的 技术 : 第 一 是 将 文本 文件 上 传 到 

# | 服务 器 ;第 二 是 读 取 服 务 器 指定 文件 夹 下 的 文本 文件 中 的 数据 ; 第 三 是 连接 数据 库 ， 将 从 文本 

到 | 文件 中 读 取 的 数据 存储 到 数据 表 中 。 
其 中 ， 最 为 关键 的 是 将 文本 文件 中 的 数据 存储 到 数据 库 中 ， 在 执行 数据 的 读 取 和 存储 时 ， 
| 有 一 个 细节 必须 把 握 。 即 应 用 addslashes() 函 数 ， 对 从 文本 文件 中 读 取 的 数据 进行 转 义 ， 因 为 在 
| 读 取 的 数据 中 可 能 包含 单 引号 (')、 双 引号 (")、 反 和 斜 线 (\) 或 NUL (NULL 字符 )， 如 果 不 
， 对 其 进行 转换 ， 则 在 执行 SQL 语句 时 可 能 会 出 现 错误 。 
| addslashes(0) 函 数 通过 反 和 斜 线 来 引用 字符 串 。 语 法 如 下 : 
string addslashes ( string str) 

| 在 其 返回 的 字符 串 中 ,为 了 数据 库 查询 语句 等 的 需要 在 某 些 特定 字符 前 加 上 了 反 和 斜 线 。 这 
，， 些 特定 字符 包括 单 引号 (')、 双 引号 (")、 反 斜 线 (\) 和 NUL (NULL 字符 )。 


| 说 明 : 
| 默认 情况 下 , PHP 指令 会 自动 对 所 有 的 GET、POST 和 COOKIE 数据 运行 转 义 ( addslashes() 
| 函数 )。 所 以 ， 如 果 是 通过 GET 或 POST 方法 提交 的 数据 是 不 需要 进行 转 义 的 ， 可 以 直接 添加 
| 到 数据 库 中 ; 如 果 提 交 的 数据 是 从 文本 文件 中 读 取 或 者 自 定义 的 内 容 ， 那 么 就 有 必要 对 其 进行 
| 转 义 。 
| 应 用 示例 
本 示例 将 文本 文件 中 的 数据 存储 到 数据 库 中 ， 其 具体 实现 步骤 如 下 : 
| (1) 创建 index.php 文件 。 首 先 ， 设 计 网 页 页 面 。 然 后 ， 创 建 一 个 表单 ， 通 过 文件 域 提交 
| 要 上 传 的 ini 文件 。 接 着 ， 获 取 表单 中 提交 的 文件 ， 将 文件 存储 到 服务 器 指定 的 文件 夹 下 ， 通 
| 过 fne get contentsO 函 数 读 取 上 传 文件 的 内 容 ， 并 且 将 读 取 的 结果 存储 到 指定 的 数据 表 中 。 最 
| 后 ， 执 行 查询 语句 ， 输 出 数据 库 中 存储 的 ini 文件 的 内 容 。 其 关键 代码 如 下 : 
| <9php 
if (! empty ( $_FILES [up_picture] [name] )) { /判断 上 传 内 容 是 否 为 空 

$type=$_FILES [up_picture] [name']; 

$types = strstr ( $type, ".' ); // 获 取 文件 后 级 

if ($types — "ini") { 

if ($_FILES [up_picture] ['eror] > 0) { // 判 断 文件 是 否 可 以 上 传 到 服务 器 
echo "上 传 错 误 :"; 
Switch ($_FILES [up_picture'] ['error']) { 
Case ll: 


echo "上 传 文件 大 小 超出 配置 文件 规定 值 "; 
break: 


case2 : 
echo "上 传 文件 大 小 超出 表单 中 约定 值 "; 
break:; 
Case 3: 
echo "上 传 文件 不 全 "; 
break: 
case 4: 
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echo "没有 上 传 文件 "; 
break:; 
} 
}else{ 
if(!is dir("./upfile/" )) { // 判 断 指定 目录 是 否 存 在 
mkdir ( "./upfile/" ); // 创 建 目 录 
} 
$file name = time 0 . strstr ($_FILES [up _picture] [name'].'.); // 定 义 上 传 文件 名 称 
$path = "./upfile/' . $file name; // 定 义 上 传 文件 名 称 和 存储 位 置 | 
下 (is_ uploaded file($ FILES [up_picture] [tmp name'] )) { | 
// 判 断 文 件 是 否 是 HTTP POST 上 传 | 
if(! move_ uploaded file($ FILES [up_picture'] [tmp_name'], $path )) { | 
// 执 行 上 传 操作 | 
echo "上 传 失 败 "; | 
}else { | 
// 读 取 文本 文件 中 的 数据 ， 并 且 实 现 转 义 和 编码 的 转换 | 
Sarr =iconv ( "gb2312", "utf-8", addslashes ( file_get_contents ( $path))): | 
include_once ("conn/conn.php"); 1/ 连接 数据 库 | 
$sql = "insert into tb_files(file name,file content,file date)values('$file name'， | 
"am . date ( "Y-m-d H:i:s" ) .")"; // 定 义 SQL 添加 语句 | 
Sresult = mysql query ( $sql, $conn ); // 执 行 添加 操作 | 
echo "文件 ，“" .$_FILES [up_picture] [name'] . ”上传 成 功 !" . "<br>"; | 
echo "大 小 : ".$_FILES [up_picture] [size] . "” 字 节 "; | 
} else { | 
echo "上 传 文件 " . $_FILES [up_pictute'] [name'] . "不 合法 ! "; | 
} | 
} else { | 
echo "上 传 文件 " . $_FILES [up_pictute'] [name'] . "类 型 不 正确 ! "; | 
} | 
} | 
?> | 


(2) 创建 conn 文件 夹 ， 编 写 conn.php 文件 ， 连 接 数 据 库 服务 器 ， 连 接 db_database 数据 | 
库 。 其 代码 如 下 : | 
<2php | 
$conn=mysql_connect("localhost", "root", "111")or die(" 连 接 数据 库 服务 器 失败 ! ". mysql_error 0 ); 

mysql select db ( "db_database". $conn ): /选择 数据 库 db_database | 
Inysql_query ( "set names utf8" ): /设置 数据 库 编码 格式 utf8 

?> | 

运行 结果 如 图 7.7 所 示 。 | 
专家 点 评 | 


将 从 文本 文件 中 读 取 的 数据 存储 到 数据 库 时 , 必须 应 用 addslashes0 函 数 对 文本 文件 返回 的 | 
字符 串 进行 转 义 ， 否 则 有 可 能 出 现 如 图 7.8 所 示 的 错误 。 
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问题 171 如何 判断 一 个 远程 的 文件 是 否 存 在 ? 


问题 讲述 


在 PHP 中 ， 如 何 判断 一 个 远程 的 文件 是 否 存在 呢 ? 


专家 解答 


判断 一 个 远程 的 文件 是 否 存在 与 访问 一 个 远程 文件 应 用 的 都 是 fopen 函数 ,如 果 通 过 fopen 


| 函数 可 以 访问 指定 的 远程 文件 ， 并 且 返 回 标识 指针 ， 则 说 明 该 远程 文件 存在 ， 否 则 说 明 该 远程 
| 文件 不 存在 。 


应 用 fopen0 函 数 判断 远程 文件 是 否 存在 ， 前 提 是 将 配置 文件 php.ini 中 的 选项 alow_url_ 


| 设置 为 ON。allow_url fopen 参数 默认 是 开启 的 , 允许 打开 HTTP 协议 和 FTP 协议 指定 的 
| 远程 文件 , 如 果 allow_url_fopen 设置 为 OFF, 则 不 允许 打开 远程 文件 , fopen0 函 数 将 返回 false。 


醒 


在 实际 的 生产 环境 中 为 了 保证 服务 器 的 安全 , php.ini 中 选项 allow_url fopen 的 值 一 般 情况 


| 下 都 设置 为 OFF， 所 以 不 支持 fopen0) 函 数 访问 远程 文件 ， 这 一 点 必须 明确 。 


问题 172 ”如 何 将 上 传 文件 转换 成 了 AR 文件 ? 


| 问题 阐述 


在 PHP 中 ， 如 何 实现 将 上 传 文件 转换 成 RAR 文件? 


专家 解答 


| 然后 是 文件 压缩 技术 。 


将 上 传 文件 转换 成 RAR 文件 ， 可 以 理解 为 两 种 技术 的 结合 应 用 ， 首 先是 文件 上 传 技术 ， 


文件 上 传 技术 自 不 必 说 ， 文 件 压 缩 技术 是 调用 com 类 中 的 mn0 方 法 ， 在 winrar 的 命令 中 


”增加 了 两 个 属性 “-ep” 将 压缩 文件 以 文件 名 的 格式 进行 存储 ， 不 包括 文件 的 路 径 ;“-df” 文 


| 件 压缩 成 功 后 删除 原始 文件 。 其 完整 的 命令 语句 如 下 : 
$obj->run ( "winrar a -ep -df " . $file path."rar" ." " . $path .", 1, false ); // 调 用 类 中 的 方法 , 文件 压缩 
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应 用 示例 
在 本 示例 中 将 上 传 的 文件 直接 以 压缩 文件 的 格式 进行 存储 。 其 具体 步骤 如 下 : | 
(1) 创建 index php 文件 。 首 先 ， 创 建 一 个 表单 ， 设 置 文 件 域 ， 提 交 上 传 文件 。 然后 设置 ， 
按钮 ， 完 成 表单 元 素 的 提交 操作 。 
(2) 在 index.php 文件 中 编写 PHP 脚本， 首先 ， 对 上 传 文件 进行 判断 ， 如 果 符 合 要 求 ， 将 | A 
上 上 传 文件 存储 到 服务 吕 指 定 文件 夹 下 。 然 后， 实例 化 PHP 中 com 类 ， 通 过 run0 方 法 执行 文件 | 
的 压缩 命令 ， 完 成 上 传 文件 的 压缩 操作 ， 并 将 原始 文件 删除 。 其 关键 代码 如 下 : | 
// 定 义 上 传 文件 名 称 和 存储 位 置 
$path ="./upfile/ . time 0 . strstr ( $_FILES [sourcefile'] [name’], '. ); 


if (is uploaded file ($ FILES [sourcefile] [tmp name']) { 
if (! move_uploaded file ( $_FILES ['sourcefile'] [tmp_ name'], $path )) { echo "上 传 失败 "; 


} else { | 
$path = realpath ( iconv ("utf-8", "gb2312", $path ) ); /获取 文件 在 服务 器 中 的 路 径 ， 
S$len = strripos ( $path, "." ); // 计 算 路 径 的 长 度 | 
$file path = substr ( $path, 0, $len ): /获取 上 传 文件 的 路 径 名 称 
$obj = new com ( "wscriptshell" ); /实例 化 com 类 


/调用 类 中 的 方法 ， 执 行文 件 压缩 和 原始 文件 删除 操作 
$obj->run ( "winrar a -ep -df " . $file path . "rar" ."". $path."", 1, false ); 
echo "文件 : ".$ FILES ['sourcefile'] [name'] . "上传 成 功 !". "<br>"; 
echo "大 小 : " . $_FILES ['sourcefile] [size] . ” 字 节 "; 
bh 
} else{ 
echo "上 传 文件 " . $_FILES ['sourcefile'] [name'] . "不 合法 ! "; 
} 


运行 本 示例 ， 选 择 要 上 传 的 文件 ， 单 击 “ 提 交 ” 按 钮 ， 其 运行 结果 如 图 7.9 所 示 。 上 传 成 


功 后 ， 可 以 查看 本 实例 根 目 录 下 的 upfile 文件 夹 ， 以 及 上 传 文件 是 否 以 压缩 文件 的 格式 进行 存 | 
储 ， 如 图 7.10 所 示 。 | 
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图 7.9 将 上 传 文件 转换 成 压缩 文件 图 7.10 上 传 后 的 文件 


问题 173 ”如 何 遍 历 、 删 除 指定 目录 下 的 所 有 文件 ? 


问题 前 述 
在 PHP 中 ， 如 何 实现 遍历 及 删除 某 一 目录 下 的 所 有 文件 呢 ? 
。231 。 到 


人 Pp 六 须知 道 的 300 个 问题 


So 
专家 解答 
1. 解决 思 
到 六 遍历 、 删 除 指定 目录 下 的 所 有 文件 关键 是 目录 、 文 件 处 理 函数 和 字符 串 处 理 函数 的 综合 运用 。 


| 目录 处 理 函 数 
(1) chdir0 函 数 ， 改 变 当前 目录 。 语 法 如 下 : 
| bool chdir(string directory) 
参数 directory 用 于 指定 更 改 后 目录 的 位 置 。 操 作成 功 返 回 True， 和 否则 返回 False。 
(2) getcewd0) 函 数 ， 获 取 当 前 工作 的 目录 。 语 法 如 下 : 
string getcwd(void): 
(3) rmdir0 函 数 ， 删 除 目录 。 语 法 如 下 : 
bool rmdir (string dirmame) 
| 尝试 删除 参数 dimame 所 指定 的 目录 。 该 目录 必须 是 空 的 ， 而 且 要 有 相应 的 权限 。 如 果 成 
| 功 则 返回 True， 失 败 则 返回 False。 
回 ”文件 操作 函数 
(1) fopen0 函 数 、fgets0 函 数 、readfile0 函 数 和 fclose() 函 数 。 
(2) filectime() 函 数 ， 返 回 指定 文件 的 索引 节点 修改 时 间 。 语 法 如 下 : 
int filectime(string filename); 
| 返回 文件 上 次 索引 节点 被 修改 的 时 间 ， 如 果 出 错 则 返回 False。 时 间 以 UNIX 时 间 戳 的 方 
| 式 返回 。 
(3) flemtime() 函 数 ， 返 回 指定 文件 flename 的 最 后 修改 时 间 。 语 法 如 下 : 
int filemtime(string filename): 
返回 文件 上 次 被 修改 的 时 间 ， 如 果 失 败 则 返回 FALSE。 时 间 以 UNIX 时 间 戳 的 方式 返 
(4) unlinkO 函 数 ， 删 除 文件 。 语 法 如 下 : 
bool unlink (string filename) 
删除 由 字符 串 filename 指定 的 文件 。 如 果 成 功 则 返回 True， 失 败 则 返回 False。 
回 “字符 串 操 作 函 数 
(1) substr0 函 数 ， 从 指定 的 字符 串 中 按照 指定 的 位 置 截取 一 定 长 度 的 字符 。 语 法 如 下 : 
string substr(string str,int start,int length) 
substr() 函 数 的 参数 说 明 如 表 7.3 所 示 。 


表 7.3 substr() 函 数 的 参数 说 明 


参数 说 明 
指定 字符 串 对 象 


加 


指定 开始 截取 字符 串 的 位 置 。 如 果 参 数 start 为 负数 ， 则 从 字符 串 的 末尾 开始 截取 
可 选 参数 ， 指 定 截取 字符 的 个 数 。 如 果 length 为 负数 ， 则 表示 取 到 倒数 第 length 个 字符 


_ J 。232 。 


第 7 章 文件 与 目录 操作 8 


Substr0 函 数 中 参数 start 的 指定 位 置 是 从 0 开始 计算 的 , 即 字符 囊 中 的 第 一 个 字符 表示 为 0。 | 


(2) strrpos0) 函 数 ， 寻 找 字 符 串 haystack 中 的 字符 needle 最 后 出 现 的 位 置 。 语 法 如 下 : | ; 
int strrpos(string haystack, char needle) | 全 | 
参数 haystack 为 寻找 的 范围 ; 参数 needle 为 寻找 的 字符 ,其 中 needle 只 能 是 一 个 字符 ， 汉 
字 则 不 可 以 。 如 果 找 不 到 指定 的 字符 ， 则 返回 False。 
(3) strtoupperO 函 数 ， 将 字符 串 str 全 部 转换 为 大 写字 符 串 。 语 法 如 下 : 
string strtoupper(string str); | 
(4) htmlentities() 函 数 ， 将 所 有 string 的 字符 都 转换 成 HTML 的 特殊 字 集 字符 串 。 语 法 | 
如 下 : | 
string htmlentities(string string); 
2. 应 用 示例 | 
本 示例 综合 运用 上 述 介绍 的 函数 ， 完 成 对 网 站 目录 的 遍历 以 及 目录 和 文件 的 删除 操作 。 其 | 
具体 步 又 如 下 : 
(1) 创建 index.php 文件 。 应 用 目录 、 文 件 处 理 函 数 和 字符 串 处 理 函 数 完成 对 指定 目录 下 | 
文件 的 读 取 操作 ， 并 设置 查看 文件 内 容 的 超 链接 和 删除 指定 文件 的 超 链接 。 在 设置 超 链接 的 参 | 
数 时 ， 应 用 urlencode0 函 数 对 字符 串 进 行 编码 。 | 
(2) 创建 look file php 文件 ， 完 成 对 指定 文件 内 容 的 读 取 操作 。 其 关键 代码 如 下 : 
<Dip 
$catalog = iconv ( "utf8", "gb2312", urldecode ( $_GET [catalog] ) ); // 获 取 文件 的 信息 ， 设 置 编码 | 


$filename = iconv ( "utf-8", "gb2312", urldecode ($_GET [filename])): /获取 文件 的 信息 ， 设 置 编码 | 
$type =iconv ( "utf-8", "gb2312", urldecode ($ GET [type] ) ); /获取 文件 的 信息 ， 设 置 编码 | 


$types = strtoupper ( $type ); // 完 成 字符 串 的 大 小 写 转换 
if ($types 一 "PHP") { // 显 示 PHP 文件 的 内 容 
$arr = file ( $catalog . "\" . $filename ): // 读 取 文 件 
foreach ( $arr as $value ) { /循环 输出 文件 内 容 
$value=htmlentities ( $value, ENT_COMPAT, "UTF-8" ): /特殊 字符 的 转换 
echo $value . "<br>": // 输 出 内 容 
} 
} else { 
$fp = fopen ( $catalog . "\\" . $filename, "1" ): // 显 示 txt 文件 的 内 容 


while ( $line = fgets ( $fp ) ) { 
$line = htmlentities ( $line, ENT_COMPAT, "GB2312" ); 。 // 特 殊 字符 的 转换 
echo iconv ( "gb2312" "utf-8", $line ): /| 输出 文件 内 容 

9 

felose ( $fp ); 


?> 


(3) 创建 delete php 文件 ， 完 成 对 指定 文件 的 删除 操作 。 
运行 本 示例 ， 实 现 目录 和 文件 的 遍历 和 删除 ， 运 行 结果 如 图 7.11 所 示 。 
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图 7.11 遍历 、 删 除 指定 目录 下 的 所 有 文件 
专家 点 评 


在 通过 URL 传递 字符 串 参 数 时 ， 应 尽量 应 用 urlencode() 函 数 对 传递 的 参数 值 进行 编码 ， 
这 样 不 但 可 以 保证 传递 参数 值 的 安全 ， 而 且 可 以 防止 传递 的 参数 值 出 现 乱 码 的 问题 。 例 如 ， 本 
实例 中 在 执行 指定 目录 、 文 件 的 删除 操作 过 程 ， 在 执行 成 功 跳 转 到 上 级 页 面 时 ， 如 果 不 对 URL 
传递 的 字符 串 参 数值 进行 编码 ， 那 么 就 无 法 完成 跳 转 的 操作 ， 同 时 会 弹出 一 个 错误 提示 。 


问题 174 ”如 何 使 用 文本 文件 统计 网 站 访问 量 ? 
问题 阐述 


网 站 的 计数 器 对 于 网 站 管理 者 来 说 是 一 个 非常 值得 关注 的 部 分 , 它 记 录 了 网 站 被 访问 的 次 
数 ， 客 观 地 反映 了 网 站 受 欢迎 的 程度 。 而 文本 计数 器 是 最 简单 的 一 种 ， 它 将 数据 存储 于 文本 文 


| 件 中 。 那 么 如 何 使 用 文本 文件 统计 网 站 访问 量 ， 并 将 其 显示 到 页 面 中 呢 ? 


| 专家 解答 


1. 解决 思路 
文本 计数 器 的 设计 思路 如 下 : 
应 用 文件 系统 函数 和 GD2 函数 ， 完 成 网 站 访问 量 的 存储 、 读 取 和 输出 。 其 实现 的 原理 如 


| 图 7.12 所 示 。 


创建 
SESSION 


直接 输出 网 站 访问 量 


判断 SESSION 
浏览 者 


是 否 为 空 
重新 输出 
网 站 访问 量 
以 写 入 方式 打开 文件 
将 数据 写 入 文件 中 


图 7.12 通过 文本 文件 统计 网 站 访问 量 设计 原理 


打开 文本 文件 
读 取 文件 中 数据 
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在 这 个 通过 文本 文件 统计 网 站 访问 量 的 程序 中 ， 概 括 起 来 应 用 了 3 个 方面 的 技术 : 
第 一 方面 ， 文 件 系统 操作 函数 ， 实 现 数据 的 写 入 、 读 取 功 能 。 
第 二 方面 ，SESSION， 通 过 SESSION 变量 防止 重复 计数 。 
第 三 方面 ，GD2 函数 ， 将 访问 量 数据 生成 图 像 ， 改 变数 据 的 输出 方式 。 
2. 应 用 示例 
本 示例 应 用 上 述 介绍 的 技术 ， 通 过 文本 文件 统计 网 站 访问 量 ， 通 过 SESSION 变量 屏蔽 页 
面 刷新 对 计数 器 的 影响 ， 通 过 GD2 函数 库 中 的 函数 输出 网 站 访问 量 。 其 具体 步骤 如 下 : | 
(1) 创建 index.php 文件 。 通 过 文件 操作 函数 和 SESSION 全 局 变量 完成 网 站 访问 量 的 统 
计 和 防止 重复 计数 ， 统 计数 据 存储 在 实例 根 目录 下 的 countertxt 中 。 通 过 img 图 像 标签 的 src 
属性 指定 gdl.php 文件 ， 以 图 像 的 形式 输出 网 站 访问 量 的 数据 。 其 关键 代码 如 下 : 
<?php session_ start(); 
if($_SESSION[temp]—""){ // 判 断 $_SESSION[temp]="" 的 值 是 否 为 空 ， 其 中 的 temp 
为 自 定义 的 变量 


if(($fp=fopen("counter.txt","r"))==false){ | 
echo "打开 文件 失败 1"; 


jelse{ 
$counter=fgets($fp,1024); // 读 取 文 件 中 数据 
fclose($fp); /关闭 文本 文件 
$counter++: /计数 器 增加 1 
$fp=fopen("countertxt","w7)): /以 写 的 方式 打开 文本 文件 <!----> 
fputs($fp,$counter); /将 新 的 统计 数据 增加 1 
felose($fp); /关闭 文件 
} 
$_SESSION[temp]=1; /登录 以 后 ，$_SESSION[temp] 的 值 不 为 空 ， 给 
$_ SESSION[temp] 赋 一 个 值 1 
<tr> 
<td height="40" align="center"><img src="gd1.php" /></td> | 
</t> | 


(2) 创建 gdl.php 文件 ， 根 据 从 文本 文件 countertxt 中 读 取 到 的 数据 ， 通 过 GD2 函数 生 
成 数据 图 像 。 其 代码 如 下 : 


<?php 
if(($fp=fopen("counter.txt","r"))=—false){ // 打 开 文件 | 
echo "打开 文件 失败 1"; | 
}else{ | 
S$counter=fgets($fp,1024): // 读 取 文 件 中 数据 | 
flose($fp); /关闭 文件 | 
Sim=imagecreate(240.24): // 通 过 GD2 函数 创建 画布 | 
S$eray=imagecolorallocate($im,255,255,255); // 定 义 背 景 颜色 | 
$color =imagecolorallocate($im,rand(0,255),rand(0,255).rand(0,255)); // 定 义 字 体 颜 色 | 
// 输 出 中 文字 符 | 
$text=iconv("gb2312","utf-8"," 网 站 的 访问 量 :"); // 对 指定 的 中 文字 符 串 进行 转换 | 
$font = "Fonts/FZHCJW.TTE"; // 定 义 使 用 字体 的 存储 位 置 | 
imagettftext($im,14,0,20,18,$color, $font, $text); // 输 出 中 文 | 
imagestring($im,5,160,5,$counter, $color); /输出 图 像 ， 输 出 网 站 的 访问 次 数 
“235。 4 
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imagepng($im); // 生 成 png 图 像 
imagedestroy($im); /销毁 图 像 
| a 
会 二 (3) 在 根 目录 下 创建 countertxt 文本 文件 ， 设 置 默认 值 为 0; 创建 fonts 文件 夹 ， 将 使 用 


的 字体 FZHCJW.TTF 存储 在 该 文件 夹 下 。 
本 示例 的 运行 结果 如 图 7.13 所 示 。 


谷 收 枉 丰 站 “ 国 设 为 言 页 “ 仿 联 系 我 们 


仿 > 绩 程 词典 在 线 服务 


吏 迎 光 伤 明 日 科 蕉 同 站 
伍 理 轧 程 词典 》 系 列 产 品 
《HP 编程 词典 》 夭 列 产品 : 
[i 人 EE 
人 本 程 主 荡 司 办》 [3 [ER 
全 编程 司 生 ) 标 稚 版 Ed 
人 久 程 司 向 】 下 大 所 soe 元 级 
得 同和 】 全 业 大 区 Ee 
《PE 护 得 滞 生 》 铅 五 版 15906 元 (2996 元 ) 3 


广 息 : 

1 。 人 jE 症 权 司 册 》 外 厂 直 价格 为 3993 元 ( 寺 三 年 网 入 设 宣 尝 习 ) ， 如果 不 午 习 网 结 课 管 ,价格 2995 元 

2 。 信 I 镶 程 全 开 辐 各》 欢 是 全 焉 护 各 局 典 》 首 及 原 
问 站 三 访问 章 : 145 


| 扶 椒 服务 才 绕 : 0431 -M4978981 B4979982 
| 图 7.13 通过 文本 文件 统计 网 站 访问 量 
| 专家 点 评 


| SESSION 防止 重复 计数 的 原理 是 : 首先 ， 在 网 页 被 访问 时 ， 初 始 化 一 个 SESSION 变量 ， 
| 并 赋 给 其 一 个 空 值 。 然 后 ， 判 断 SESSION 变量 的 值 是 否 为 空 ， 如 果 为 室 ， 则 将 计数 器 的 值 增 
| 加 1， 并 且 为 SESSION 变量 赋值 为 1。 此 时 ， 在 当前 页 中 ，SESSION 变量 的 值 已 经 不 为 空 ， 
无 论 如 何 刷新 ，SESSION 变量 的 值 都 不 会 改变 ， 所 以 计数 器 的 值 也 不 会 增加 。 
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香 理 有 昔 有 昔 理 有 昔 昔 理 理 吾 玉 


问题 175 
问题 176 
问题 177 
问题 178 
问题 179 
问题 180 
问题 181 
问题 182 
问题 183 
问题 184 
问题 185 


为 什么 要 面向 对 象 ? 

构造 方法 和 析 构 方法 有 什么 不 同 ? 
为 什么 要 封装 ? 

如 何 继承 一 个 类 ? 

抽象 类 与 接口 的 区 别 是 什么 ? 

如 何 封 装 一 个 项 目 工具 类 ? 

如 何 将 购物 车 中 的 功能 封装 到 购物 车 类 中 ? 
如 何 制作 柱 形 图 绘制 类 ? 

如 何 制作 折线 图 绘制 类 ? 

如 何 制 作 饼 形 图 给 wa 

如 何 封 装 验 证 码 类 


lr 


世 


2) 人 Pp 六 须知 道 的 5300 个 问题 


问题 175 ”为 什么 要 面向 对 象 ? 


问题 阐述 
面向 对 象 (OOP) 的 编程 方式 是 PHP 的 突出 特点 之 一 ， 那 么 为 什么 要 使 用 面向 对 象 呢 ? 


专家 解答 


| 别 进行 介绍 。 


要 解释 为 什么 使 用 面向 对 象 ， 首 先 应 知道 面向 对 象 思想 的 来 源 和 它 能 解决 的 问题 ， 下 面 分 


1. 面向 对 象 思想 的 来 源 
面向 对 象 思想 来 源 于 对 现实 世界 的 认 知 。 现实 世界 缤纷 复杂 、 种 类 繁多 , 难于 认识 和 理解 。 


| 但 是 聪明 的 人 们 学 会 了 把 这 些 错综复杂 的 事物 进行 分 记 包 注 车 


| 类， 从 而 使 世界 变 得 井井有条 。 例 如 ， 现 实生 活 中 ， 由 EA MK 


| 各 式 各 样 的 汽车 抽象 出 汽车 的 概念 ， 由 形形色色 的 鸟 抽 所 
| 象 出 鸟 的 概念 ， 由 五 彩 斑 阐 的 鲜花 抽象 出 花 的 概念 等 ， 着 


人 人 


| 如 图 8.1 所 示 。 图 8.1 面向 对 象 思想 来 源 于 对 现实 


2. 面向 对 象 能 解决 的 问题 世界 的 认 知 

通过 面向 过 程 进行 Web 项 目 开发 ， 一 般 都 会 存在 以 下 问题 : 

加 ”代码 重用 性 差 

程序 开发 过 程 中 ， 有 时 会 遇 到 实现 相同 功能 的 代码 到 处 进行 复制 、 粘 贴 的 情况 ， 存 在 代码 


| 重用 性 差 的 问题 。 


加 ”项 目 可 维护 性 差 
项 目 开发 过 程 中 ， 项 目的 可 读 性 、 可 修改 性 和 可 测试 性 是 项 目的 重要 质量 指标 ， 用 传统 方 


| 法 开发 出 来 的 项 目 ， 维 护 费用 和 成 本 比较 高 。 


回 开发 出 来 的 项 目 不 能 满足 用 户 的 需求 变化 
用 传统 的 面向 过 程 方法 开发 出 的 项 目 涉及 各 种 不 同 领域 的 知识 , 在 开发 需求 模糊 或 需求 动 


| 态 变化 的 系统 时 ， 所 开发 项 目 往 往 不 能 满足 用 户 需求 。 


而 面向 对 象 正 是 解决 以 上 这 些 问题 的 一 种 方法 ， 可 以 对 大 量 零 散 代码 进行 有 效 组 织 ， 从 而 


| 使 PHP 具备 大 型 Web 项 目 开发 的 能 力 ， 同 时 还 可 以 提高 网 站 的 易 维护 性 和 易 读 性 。 
| 专家 点 评 


面向 对 象 是 PHP 编程 的 核心 技术 ， 它 相对 于 以 往 的 面向 过 程 编程 来 说 ， 使 得 程序 的 设计 更 


| 加 合理 , 而 且 易 于 维护 , 所 以 在 开发 程序 时 ,建议 大 家 将 面向 对 象 的 编程 思想 应 用 到 项 目 开 发 中 。 


问题 176 ”构造 方法 和 析 构 方法 有 什么 不 同 ? 


| 问题 阐述 


程序 中 使 用 类 时 有 两 种 特殊 的 方法 ， 分 别 是 构造 方法 和 析 构 方法 ， 它 们 有 什么 不 同 呢 ? 
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专家 解答 
构造 方法 和 析 构 方法 都 具有 与 类 相同 的 名 称 ， 下 面 分 别 对 它们 进行 介绍 。 
1， 构 造 方法 


构造 方法 是 对 象 创建 完成 后 第 一 个 被 对 象 自动 调用 的 方法 ， 它 存在 于 每 个 声明 的 类 中 , 是 | 
一 个 特殊 的 成 员 方法 。 如 果 在 类 中 没有 直接 声明 构造 方法 ， 那 么 类 中 会 默认 生成 一 个 没有 任何 
参数 且 内 容 为 空 的 构造 方法 。 

在 PHP 中 , 构造 方法 的 声明 有 两 种 情况 : 第 一 种 在 PHP5 以 前 的 版 本 中 ,构造 方法 的 名 称 | 
必须 与 类 名 相同 ;第 二 种 在 PHP5 的 版 本 中 ， 构 造 方法 的 方法 名 称 必须 是 以 两 个 下 划 线 开始 的 
“_construct0”。 虽然 在 PHP5 中 构造 方法 的 声明 方法 发 生 了 变化 ， 但 是 以 前 的 方法 仍然 是 可 | 
用 的 | 


PHP5 中 的 这 个 变化 是 考虑 到 构造 函数 可 以 独立 于 类 名 ， 当 类 名 发 生变 化 时 不 需要 修改 相 
应 的 构造 函数 的 名 称 。 通 过 _constructO 声 明 构 造 方法 的 语法 如 下 : | 


function construct([mixed args [,...]){ 
/方法 体 
} 


在 PHP 中 ， 一 个 类 只 能 声明 一 个 构造 方法 。 在 构造 方法 中 可 以 使 用 默认 参数 ， 实 现 其 他 面 | 
向 对 象 的 编程 语言 中 构造 方法 重 载 的 功能 。 如 果 在 构造 方法 中 没有 传 入 参数 ， 那 么 将 使 用 默认 | 
参数 为 成 员 变 量 进 行 初始 化 。 
例如 ， 使 用 _construct0 声 明 一 个 与 类 名 不 同 的 构造 方法 。 其 代码 如 下 : 
> | 
构造 函数 : 当 类 被 实例 化 后 构造 函数 自动 执行 。 所 以 如 果 用 户 希 望 在 实例 化 的 同时 调用 某 个 方 | 
法 可 以 把 此 方法 通过 this 关键 字 调用 。 | 
class mysql{ /定义 类 名 称 
var $localhost: /定义 变量 
Var $name; 
Var $pwd; 
Var $db; 
Var $conn; 
public function __construct($localhost, $name.,$pwd.,$db){ // 构 造 函 数 
$this->localhost=$localhost; 
S$this->name=$name; 
S$this->pwd=$pwd; 
S$this->db=$db; 
S$this->connect(; 


} 
/省 略 了 部 分 代码 


» 

$msl=new mysql("127.0.0.1","root"."111","db_database"): // 实 例 化 对 象 

Smsl->GetIdO; // 对 象 句柄 调用 指定 的 方法 
?> 


2.， 析 构 方法 | 
析 构 方法 的 作用 和 构造 方法 正好 相反 , 它 是 对 象 被 销毁 之 前 最 后 一 个 被 对 象 自动 调用 的 方 | 
as 239is Le 
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法 。 该 方法 是 PHP5S 中 新 添加 的 内 容 ， 实 现在 销毁 一 个 对 象 之 前 执行 一 些 特定 的 操作 ， 诸 如 关 
| 闭 文件 、 释 放 内 存 等 。 

| 析 构 方法 的 声明 格式 与 构造 方法 类 似 ， 都 是 以 两 个 下 划 线 开头 的 “_destruct”， 析 构 函 数 


会 内 | 没有 任何 参数 。 其 语法 格式 如 下 : 
sw | function destructO{ 
7 方法 体 ， 通 常 是 完成 一 些 在 对 象 销毁 前 的 清理 任务 


} 


| 在 PHP 中 有 一 种 “垃圾 回收 ”机 制 ， 可 以 自动 清除 不 再 使 用 的 对 象 ， 从 而 释放 内 存 。 而 
| 析 构 方法 就 是 在 这 个 垃圾 回收 程序 执行 之 前 被 调用 的 方法 ， 在 PHP 中 它 属于 类 中 的 可 选 内 容 。 


| 专家 点 评 


| 通过 上 面 的 学 习 可 以 看 出 ,构造 函数 和 析 构 函数 是 完全 不 同 的 两 个 概念 ， 构 造 函 数 用 来 初 
| 始 化 新 对 象 ， 并 为 对 象 分 配 内 存 ， 而 析 构 函数 用 来 进行 垃圾 回收 。 


问题 177 为 什么 要 封装 ? 


| 问题 阐述 
| 面向 对 象 编程 中 的 一 个 重要 特征 就 是 封装 ， 那 么 ， 为 什么 要 实现 封装 呢 ? 
| 专家 解答 


| 面向 对 象 编程 的 特点 之 一 是 封装 性 ， 将 类 中 的 成 员 属性 和 方法 结合 成 一 个 独立 的 相同 
| 单位 ， 并 尽 可 能 隐藏 对 象 的 内 容 细节 。 其 目的 就 是 确保 类 以 外 的 部 分 不 能 随意 存 取 类 的 内 
| 部 数据 (成 员 属性 和 成 员 方 法 )， 从 而 有 效 避 免 外 部 错误 对 类 内 数据 的 影响 。 

| 此 外 ， 封 装 还 可 以 解决 数据 存 取 的 权限 问题 ， 可 以 使 用 封装 将 数据 隐藏 起 来 形成 一 个 封闭 
| 的 空间 ， 然 后 可 以 设置 哪些 数据 只 能 在 这 个 空间 中 使 用 ， 哪 些 数据 可 以 在 空间 外 部 使 用 。 如 果 
| 一 个 类 中 包含 敏感 数据 ， 有 些 人 可 以 访问 ， 有 些 人 不 能 访问 ， 如 果 不 对 这 些 数据 的 访问 加 以 限 
| 制 ， 后 果 将 会 非常 严重 。 所 以 在 编写 程序 时 ， 要 对 类 的 成 员 使 用 不 同 的 访问 修饰 符 ， 从 而 定义 
| 它们 的 访问 级 别 。 

| 类 的 封装 可 以 通过 关键 字 public、private、protected、static 和 final 来 实现 。 

应 用 示例 

| 在 本 示例 中 通过 private (私有 成 员 ) 定义 私有 变量 ， 然 后 对 变量 进行 修改 与 访问 ， 并且 直 
| 接 调用 私有 变量 ， 看 会 有 什么 情况 发 生 。 其 代码 如 下 : 


<2php 
class Car{ 
Private $carName=" 奥 迪 系 列 ": /定义 私有 变量 并 赋值 
public function setName($carName){ // 利 用 set 方法 为 设置 变量 值 
S$this->carName=$carName; 
public function getNameO{ /利用 get 方法 返回 变量 值 
Tetum $this->carName: 
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} 

class SmallCar extends Car{ /继承 

} 

S$car=new SmallCar0: // 实 例 化 子 类 对 象 
$car->setName("Q7"); /为 子 类 变量 赋值 
echo "正确 操作 私有 变量 <br>"; 

echo $car->getName(); // 输 出 子 类 变量 的 值 


echo "<br> 错 误 操作 私有 变量 "; 
echo Car::$carName; 
> 


运行 结果 如 图 8.2 所 示 。 
专家 点 评 


作 坟 有 变量 


对 于 成 员 方法 ， 如 果 没有 写 关键 字 ， 那 么 默认 就 是 | 只 FRRRRaRSRRR ， 


public。 为 了 建立 一 个 良好 的 编程 习惯 ， 建 议 读者 在 定义 。 一 ee 
方法 与 变量 时 都 加 上 关键 字 。 ee 


问题 178 ”如 何 继承 一 个 类 ? 


问题 阐述 
如 何 继承 一 个 类 ? 
专家 解答 | 
继承 是 PHP5 面向 对 象 程序 设计 的 重要 特性 之 一 ， 类 的 继承 就 是 指 子 类 能 够 共享 父 类 成 员 | 
变量 和 成 员 方法 的 一 种 机 制 ， 这 是 类 与 类 之 间 的 一 种 关系 的 体现 。 通 过 类 继承 机 制 ， 可 以 利用 | 
已 有 的 父 类 来 派生 新 的 子 类 ， 子 类 不 仅 可 以 拥有 新 定义 的 成 员 变 量 和 成 员 方 法 ， 而 且 还 同时 拥 | 
有 父 类 所 有 的 成 员 变 量 和 成 员 方法 。 | 
在 PHP 中 ， 要 实现 类 的 继承 可 以 使 用 关键 字 extends， 其 语法 格式 如 下 : 


class Parentl { /定义 父 类 Parentl 
// 父 类 的 成 员 

} 

class Child extends Parent] { // 定 义 子 类 Child 
// 子 类 的 成 员 

} 


通过 上 述 继承 关系 即 可 在 子 类 中 使 用 parent 关键 字 加 “::” 操作 符 调用 父 类 的 公有 方法 或 | 
属性 ， 或 者 重 写 非 final 关键 字 修饰 的 方法 。 | 
应 用 示例 
在 本 示例 中 ， 首 先 定义 父 类 水 果 类 ， 并 在 该 类 中 定义 获得 水 果 颜色 和 水 果 形状 的 方法 ， 然 | 
后 定义 水 果 类 的 子 类 苹果 类 ， 在 苹果 类 中 只 定义 构造 方法 用 于 实现 对 类 进行 初始 化 ， 最 后 通过 | 
调用 父 类 的 方法 获得 苹果 的 颜色 和 形状 。 其 具体 步骤 如 下 : 
(1) 定义 水 果 类 Fruit, 在 该 类 中 定义 颜色 属性 $color 和 形状 属性 $shape， 并 通过 构造 方法 | 
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对 这 两 个 属性 进行 初始 化 ， 然 后 定义 用 于 获得 这 两 个 属性 的 getColor0 方 法 和 getShape() 方 法 。 


| 代码 如 下 所 示 : 
| class Fruit { 
| /定义 水 果 类 
食 关 | | private $color; /颜色 属性 
| Private $shape; // 形 状 属性 
public function __ construct ($color, $shape) { // 构 造 方法 对 水 果 类 初始 化 


S$this->color = $color; 
$this->shape = $shape; 


} 

public function getColor | { // 获 得 水 果 颜 色 
Tetum $this->color; 

} 

public function getShape | { // 获 得 水 果 形 状 
return $this->shape; 

} 


} 
| (2) 定义 苹果 类 Apple， 使 该 类 继承 自 水 果 类 Frmuit， 并 定义 苹果 类 的 构造 方法 ， 在 构造 
| 方法 中 使 用 parent 关键 字 调 用 父 类 构造 方法 对 父 类 进行 初始 化 。 代 码 如 下 所 示 : 
| class Apple extends Fruit ={ // 苹 果 类 继承 水 果 类 


public function construct ($color, $shape) { // 构 造 方法 对 类 进行 初始 化 
parent::_ construct($color, $shape); 


} 

| (3) 建立 苹果 颜色 和 形状 录入 表单 。 用 户 在 表单 中 输入 苹果 的 颜色 和 形状 后 单 击 “ 提 交 ” 
| 按钮 ， 将 通过 如 下 代码 打印 苹果 的 颜色 和 形状 属性 。 
| ifisset($_ POST['color]) && $_ POST['color']!="){ /判断 是 否 已 经 提交 表单 

require 'Fruit.php'; // 包 含水 果 类 

$apple = new Apple($_POST['color],$_POST['shape]): 。“”// 对 水 果 类 进行 实例 化 

echo '<font color="blue"> 我 看 见 的 苹果 是 : '.$apple->getColor().',".$apple->getShape0.' 的 </font>'; 

/打印 苹果 属性 


} 
| 上 述 代码 首先 判断 用 户 是 否 已 经 提交 表单 ， 如 果 是 ， 则 首先 使 用 require 语句 包含 苹果 类 ， 
| 然后 使 用 new 关键 字 对 苹果 类 进行 实例 化 , 最 后 使 用 苹果 类 
| 实例 的 对 象 调用 父 类 中 的 getColor0 方 法 和 getShape() 方 法 
”打印 苹果 的 属性 。 

| 运行 本 示例 ， 结 果 如 图 8.3 所 示 。 首 先 在 表单 中 输入 苹 
| 果 的 颜色 和 形状 属性 ， 然 后 单 击 “ 提 交 ” 按 钮 即 可 在 页 面 中 
| 打印 出 苹果 的 颜色 和 形状 属性 。 图 83 打印 苹果 颜色 和 形状 属性 


| 专家 点 评 
| 本 示例 中 已 经 说 明 父 类 中 使 用 final 关键 字 所 修饰 的 方法 不 能 被 重 写 ， 那 么 如 何 使 一 个 类 
| 不 能 被 其 他 类 所 继承 呢 ? 同样 可 以 通过 final 关键 字 来 修饰 ， 如 下 所 示 : 


final class Fruit ”// 使 用 final 关键 字 修饰 类 { 
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上 述 的 Fruit 类 被 final 关键 字 所 修饰 ， 这 样 该 类 就 不 能 被 其 他 类 所 继承 。 
问题 179 ”抽象 类 与 接口 的 区 别 是 什么 ? 


问题 前 述 
抽象 类 与 接口 的 区 别 是 什么 ? 
专家 解答 
1.， 抽象 类 | 
抽象 类 是 一 种 不 能 被 实例 化 的 类 ， 只 能 作为 其 他 类 的 父 类 来 使 用 。 抽 象 类 使 用 abstract 关 | 
键 字 来 声明 ， 其 语法 格式 如 下 : | 
abstract class 抽象 类 名 称 { 


/抽象 类 的 成 员 变量 列表 
abstract function 成 员 方法 1( 参数 ); /定义 抽象 方法 
abstract function 成 员 方法 2( 参数 ); /定义 成 员 方法 


} | 
抽象 类 和 普通 类 相似 ， 包 含 成 员 变量 、 成 员 方法 。 两 者 的 区 别 在 于 ， 抽 象 类 至 少 要 包含 一 
个 抽象 方法 。 抽 象 方法 没有 方法 体 ， 其 功能 的 实现 只 能 在 子 类 中 完成 ， 其 也 是 使 用 abstract 关 | 
键 字 来 修饰 。 | 
注意 : 

在 抽象 方法 后 面 要 有 分 号 “:”。 


抽象 类 和 抽象 方法 主要 应 用 于 复杂 的 层次 关系 中 , 这 种 层次 关系 要 求 每 一 个 子 类 都 包含 并 | 
重 写 某 些 特定 的 方法 。 例 如 ， 中 国 的 美食 是 多 种 多 样 的 ， 有 吉 菜 、 鲁 菜 、 川 菜 、 粤 菜 等 。 每 种 | 
菜系 使 用 的 都 是 前 、 炒 、 京 、 炸 等 手法 ， 只 是 在 具体 的 步骤 上 各 有 各 的 不 同 。 如 果 把 中 国美 食 | 
当 作 一 个 大 类 Cate， 下 面 的 各 大 菜系 就 是 Cate 的 子 类 ， 而 前 、 炒 、 京 、 炸 则 是 每 个 类 中 都 有 | 
的 方法 。 每 个 方法 在 子 类 中 的 实现 都 是 不 同 的 ， 在 父 类 中 无 法 规定 。 为 了 统一 规范 ， 不 同 子 类 | 
的 方法 要 有 一 个 相同 的 方法 名 : decoct ( 煎 )、stir_ fry ( 炒 )、cook( 毫 )、fry ( 炸 )。 

2. 接口 | 

接口 的 作用 是 实现 PHP 的 多 重 继 承 。 接 口 类 通过 interface 关键 字 来 声明 ， 接 口中 声明 的 | 
方法 必须 是 抽象 方法 ， 接 口中 不 能 声明 变量 ， 只 能 使 用 const 关键 字 声 明 为 常量 的 成 员 属 性 ， | 
并 且 接 口中 所 有 成 员 都 必须 具备 public 的 访问 权限 。 接 口 声 明 的 语法 格式 如 下 : 


interface 接口 名 称 { // 使 用 interface 关键 字 声 明 接 口 
// 常 量 成 员 // 接 口中 成 员 只 能 是 常量 
/抽象 方法 ; /成 员 方法 必须 是 抽象 方法 

} 


接口 和 抽象 类 相同 都 不 能 进行 实例 化 的 操作 ， 也 需要 通过 子 类 来 实现 。 但 是 接口 可 以 直接 
使 用 接口 名 称 在 接口 外 去 获取 常量 成 员 的 值 。 | 
例如 ， 下 面 声明 一 个 One 接口 ， 其 代码 如 下 : 
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interface One{ /声明 接口 
const CONSTANT='CONSTANT value': /声明 常量 成 员 属 性 
function FunOne(); /声明 抽象 方法 
| } 
全 | ”接口 之 则 也 可 以 实现 继承 ， 同 样 需要 使 用 extends 关键 字 。 
| 例如 ， 下 面 声 明 一 个 Two 接口 ， 通 过 extends 关键 字 继 承 One。 其 代码 如 下 : 
interface Two extends Onef /声明 接口 ， 并 实现 接口 之 间 的 继承 
| fonction FunTwo0; /声明 抽象 方法 
} 


抽象 类 和 接口 既 有 相同 之 处 ， 也 有 类 似 的 功能 ， 但 是 它们 之 间 还 是 存在 很 多 区 别 的 : 
抽象 类 的 操作 通过 继承 关键 字 extends 来 实现 , 而 接口 的 使 用 则 通过 implements 关键 
字 来 实现 。 

抽象 类 中 有 数据 成 员 ， 可 以 实现 数据 的 封装 ， 但 是 接口 没有 数据 成 员 。 

抽象 类 中 可 以 有 构造 函数 ， 但 是 接口 没有 构造 函数 。 

抽象 类 的 方法 可 以 通过 private、protected 或 public 关键 字 来 修饰 ， 而 接口 中 的 方法 
只 能 使 用 public 来 修饰 。 

一 个 类 只 能 继承 于 一 个 抽象 类 ， 而 一 个 类 可 以 同时 实现 多 个 接口 。 

抽象 类 中 可 以 有 成 员 方 法 的 实现 代码 ， 而 接口 中 不 可 以 有 成 员 方 法 的 实现 代码 。 


问题 180 ”如 何 封装 一 个 项 目 工 具 类 ? 


加 加 网 


加 辐 


| 问题 阐述 


| 封装 项 目 工具 类 是 将 项 目 中 某 些 特殊 的 功能 封装 到 一 个 类 中 , 在 项 目 需 要 使 用 这 些 特殊 功 
| 能 时 ， 直 接 调 用 类 中 的 方法 即 可 。 在 这 个 类 中 ， 封 装 的 方法 包括 : 获取 网 站 的 相对 路 径 、 货 币 
| 的 格式 化 、HTML 标记 的 转 义 输出 、 中 文字 符 串 的 截取 、 加 载 FCKEditor 在 线 编辑 器 、 输 出 
| FCKEditor 编辑 器 的 编辑 结果 以 及 关键 字 描 红 。 


| 专家 解答 
| /六 六 玉米 六 六 六 六 六 六 率 六 六 六 六 六 六 玉 六 六 六 六 六 六 六 六 浆 闵 六 六 六 六 六 六 六 六 六 六 六 六 浆 闵 六 六 六 六 六 六 六 六 六 六 率 六 六 六 六 六 六 六 六 六 六 冰冰 六 
* @ 说 明 : Util 项 目 工具 类 ， 可 进行 货币 格式 化 、_HIML 转 义 、 中 文字 符 串 截取 及 关键 字 描 红 等 操作 
*# @ 作者 : 明日 科技 
*@ 学 习 社区 : www.mrbccd.com 
* @ E-mail:mingrisoft(Wmingrisoft.com 
dg ed ed ee de 
class Util{ 
Var $arrayIni; 
/** 


* 构造 方法 
* @retumn Util 
*/ 


function Util Of 
Sthis->arrayIni = parse_ini file('config/lzhConfig.ini"); 
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/* 
*@ 方法 说 明 : 
* ”网 站 相对 地 址 
Sy 
function baseUrl |O{ 
SarrayIni = $this->arrayIni; 
Tetum $arrayIni["baseUrl']; 
由 | 
让 | 
* @ 方法 说 明 : | 
* ”对 货币 进行 格式 化 
A | 
* @ 参数 说 明 : 
* ”Smoney: 金额 
有 
function moneyFormat ($money){ 
Teturn str_replace(',', ", number format($money, 2)); 
> 
*@ 方法 说 明 : 
* ”对 html 标记 进行 转 义 输出 
六 
*@ 参数 说 明 : 
* ”Stext: 需 转 义 的 文本 
function unHtml ($text){ 
S$str = htmlspecialchars($text); 
$str = str_replace(chr(13), "<br>", $str); 
$str = nl2br($str); 
$str = str_replace(chr(32), "&nbsp;", $str); 
return $str; 


es 

* @ 方法 说 明 : 

* ”对 中 文 进行 截取 ， 防 止 出 现 乱 码 

六 

* @ 参数 说 明 : 

* ”$str: 要 截取 的 字符 串 | 

* Sstart: 开始 截取 的 位 置 | 

* ”Slen: 截取 的 长 度 

*/ | 

function msubstr ($str, $start, $len){ 
Stmpstr=mb_substr($str,$start, $len,"utf-8"); | 
Tetum $tmpstr; 

} | 

Us | 

*@ 方法 说 明 : | 

* ”集成 FCKEditor | 


*@ 参数 说 明 : 
* ”Sname: 编辑 器 名 称 | 
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* ”Svalue: 编辑 器 内 容 
| * ”S$width: 编辑 器 宽度 
| * ”Sheight: 编辑 器 高 度 
| 3 
| fonction editor ($name, $value, $width ="'100%', $height ='200"){ 
食 生 | | Tequire_once 'library/fckeditor/fckeditor.php'; 
| $arrayIni = $this->arrayIni; 
$oFCKeditor = new FCKeditor($name); 
| $oFCKeditor->BasePath = $arrayIni[baseUrl] . "/library/fckeditor/"; 
| $oFCKeditor->Width = $width: 
| S$oFCKeditor->Height = $height: 
| S$oFCKeditor->ToolbarSet = 'Default': 
S$oFCKeditor->Value = $value: 
| S$oFCKeditor->Create(); 


/* 
*@ 方法 说 明 : 
* “对 编辑 器 输出 内 容 转 义 
* 
* @ 参数 说 明 : 
* ”Stext: 输出 内 容 
#/ 
function editorUnHtml ($text){ 
Teturm stripslashes($text); 
} 


/* 
* @ 方法 说 明 : 
* ”对 关键 字 进 行 描 红 
六 
* @ 参数 说 明 : 
* ”S$arrayKey: 关键 字 
* ”$text: 查询 范围 
hah 
function setRed ($arrayKey, $text){ 
S$tmpText = $this->unHtml($text); 
for ($i= 0; $i < count($arrayKey): $i1++) { 
if ($arrayKey[$i] (=") { 
| S$tmpText = str_Ieplace(trim($arrayKey[$i]), "<font color=\"#FF0000\">".$arrayKey 
| [$1] . "</font>", StmpText): 
| } 


| Tetum $tmpText; 
| } 
应 用 示例 
在 应 用 Util 项 目 工具 类 时 ， 首 先进 行 类 的 实例 化 ， 然 后 通过 实例 化 返回 的 对 象 调用 类 中 的 
方法 ， 实 现 对 应 的 功能 。 类 实例 化 的 代码 如 下 ; 
| S$util=new Util0; /实例 化 项 目 工具 类 
| 通过 Utl 关中 的 baseUal( 方 法 获取 网 站 中 的 相对 地 址 。 
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<div style= "width:930px; height:21px: backeround:-url(<?php echo Sutil->baseUrl; | 
border-top:1px solid #CCCCCC; padding-top:Spx: text-align:left; padding-left:20px: font-size:13px"> 
通过 Util 类 中 的 moneyFormat0 方 法 对 货币 进行 格式 化 。 

<?php echo Sutil->moneyFormat($res[$i]["m price"]):?> 
通过 Util 类 中 的 unHtml0 方 法 对 HTML 标记 进行 转 义 输出 。 
<?php echo S$util->unHtml($res[S$il['area"]):?> 
通过 Util 类 中 的 msubstr0 方 法 对 字符 串 进 行 截取 。 
<?php echo S$util->msubstr($res[$i]['name'],0,4);?> 
通过 Util 类 中 的 editor0 方 法 集成 FCKEditor 在 线 编辑 器 。 

<?php echo Sutil->editor ('content', ") :?> 
通过 Util 类 中 的 setRed() 方 法 对 指定 的 关键 字 进 行 描 红 。 此 时 对 字符 串 中 的 “00” 进 行 描 红 。 

<?php echo $util->setRed (‘00',$util->msubstr($res[$i]['addtime'],0,4));?> 


问题 181 如何 将 购物 车 中 的 功能 封装 到 购物 车 类 中 ? 


问题 阐述 

购物 车 的 主要 功能 是 保留 用 户 选择 的 商品 信息 。 用 户 可 以 在 购物 车 内 设置 选 购 商 品 的 数 
量 、 显 示 选 购 商 品 的 总 金额 ， 同 时 还 可 以 执行 清除 选择 的 全 部 商品 信息 、 重 新 选择 商品 信息 等 
操作 。 为 了 提高 开发 的 效率 ， 如 何 将 购物 车 中 的 各 种 功能 都 封装 到 Cart 购物 车 类 中 ,在 需要 各 
种 功能 时 ， 只 需 调用 类 中 对 应 的 方法 即 可 ? 


专家 解答 
/六 迷 玉 六 六 六 来 率 闵 浆 率 六 六 这 六 六 率 来 冰 闵 六 六 六 六 六 冰 浆 闵 率 浆 率 率 闵 闵 率 六 六 率 六 六 六 闲 六 冰 亲 六 闲 闵 素 率 闲 闵 率 六 六 六 六 冰 六 冰冰 冰冰 六 冰冰 
*@ 说 明 : 购物 车 类 
*@ 作者 : 明日 科技 
*@ 学 习 社区 : www.mrbccd.com 
* @ E-mail:mingrisoft@mingrisoft.com 
玉 玉 六 六 六 六 六 六 六 六 六 六 来 六 六 六 六 冰 素 六 六 率 闵 六 六 六 六 率 六 六 六 六 六 率 六 六 六 六 六 素 六 六 六 六 六 六 六 六 六 六 六 六 六 六 闵 六 六 六 六 六 六 六 六 六 六 六 / 
class Cart{ 
var $idStr: /商品 ID 组 成 的 字符 串 ， 用 字符 @ 进 行 连接 
var $numStr; /商品 数量 组 成 的 字符 串 ， 用 字符 @ 进 行 连接 


/* 
*@ 方法 说 明 
* ”构造 方法 


* @ 参数 说 明 : 
* ”$idStr: 存储 商品 DD 的 字符 串 
* ”$numStr: 存储 商品 数量 的 字符 串 
#/ 
function Cart ($idStr, $numStr) 
{ 
S$this->idStr = $idSstr /为 ID 串 赋 初 值 
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S$this->numStr = $numstr /为 数量 串 赋 初 值 


| * @ 方法 说 明 : 
会 内 | * “将 商品 添加 到 购物 车 
! 六 
| * @ 参数 说 明 : 
* ”S$goodsId: 商品 也 
| * ”S$goodsNum: 购买 数量 
| function addCart ($goodsId, $goodsNum) 
| { 
| $arrayIds = explode(@', $this->idStr); 
| Sflag = 0; 
for ($i= 0; $1 < count($arrayIds); $1 ++) { 
if ($arrayIds[$i] 一 $goodsId) { 
Sflag= 1; 
b 


} 
if (Sflag —0){ 
S$this->idStr = $goodsId . '@'; 
S$this->numStr .= $goodsNum . '@'; 
} else { 
echo "<script>alert(' 该 商品 已 经 添加 ! ");</script>"; 
) 


} 
/本 
* @ 方法 说 明 : 
* 。 将 商品 从 购物 车 中 移出 
* 
* @ 参数 说 明 : 
* ”$goodsId: 商品 了 DD 
function removeCart ($goodsId) 
{ 


S$arraylds = explode(@'. $this->idStr); 
$arrayNums = explode('@', $this->numStr); 
for ($i= 0; $i < count($arrayIds): $1 ++) { 

| if ($arrayIds[$i] 一 $goodsId) { 

| unset($arrayIds[$i]): 

| unset($arrayNums[$i]): 

| } 


b 

Sthis->idStr = implode('@', $arrayIds): 

| S$this->numStr = implode('@', $arrayNums); 
} 

A 

| *@ 方法 说 明 : 

| * ”更 改 购物 车 中 的 商品 数量 

| * @ 参数 说 明 : 

| * ”S$goodsId: 商品 ID 

| * ”S$goodsNum: 购买 数量 
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i 
function changeNum ($goodsId, $goodsNum) 
{ 
$arrayIds = explode('(@', $this->idStr); 
S$arrayNums = explode('@', $this->numStr); 
for ($i = 0; $i < count($arrayIds); $i1++) { 
if ($arrayIds[$i] 一 $goodsId) { 
$arrayNums[$i] = $goodsNum; 
} ! 
} | 
S$this->idStr = implode((@O', $arrayIds): | 
S$this->numStr = implode('@', $arrayNums); | 
A 
*@ 方法 说 明 : 
* ”清空 购物 车 
function setCartNull 0 


$this->idStr ="; 
S$this->numStr ="; 
> 
*@ 方法 说 明 : 
* ”获取 购物 车 中 商品 字符 串 
function getIdStr 0 
{ 


2 
* @ 方法 说 明 : 
* 获取 购物 车 中 商品 数量 字符 串 
dl 

function getNumStr 0 

{ 


9 

} 
应 用 示例 | 
明日 科技 书店 作为 一 个 电子 商务 类 网 站 ， 购 物 车 功能 是 必 不 可 少 的 。 下 面 笔者 就 以 明日 科 | 
技 书店 为 背景 ， 介 绍 如 何 应 用 购物 车 类 实现 购物 车 功能 。 其 具体 步骤 如 下 : | 
(1) 购物 车 中 商品 的 增 、 删 、 改 等 业务 逻辑 操作 都 是 在 cartphp 文件 中 实现 的 , 为 了 区 分 | 
不 同 的 商品 购买 者 ， 可 以 应 用 $_SESSION 全 局 数组 来 保存 商品 ID 串 和 商品 数量 串 ， 所 以 在 | 
cart.php 文件 的 代码 前 部 应 分 别 对 保存 商品 ID 串 和 商品 数量 串 的 9_SESSION 数组 元 素 进行 初 | 

始 化 设置 。 其 具体 实现 代码 如 下 : 

这 ! isset($_SESSION[idstr])ll'isset($_SESSION[mumStr])1U 判 断 是 否 设置 商品 ID 的 SESSION 变量 


$_SESSION['idstr] ="; /如 果 没 有 ， 则 设置 二 者 ， 并 赋 空 值 
$_SESSION['numStr] = 


return $this->idStr: 


retum $this->numStr; 
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名 R 
} 
$cart = new Cart($_SESSION['idStr], $_SESSION['mumStr]): /实例 购物 车 类 


上 述 代 码 首 先 判 断 用 于 保存 商品 ID 串 的 $_SESSION['idStr] 数 组 元 素 和 保存 商品 数量 串 的 
$_SESSION[mumstr] 数 组 元 素 是 否 已 经 被 设置 ， 如 果 没 有 被 设 定 ， 则 说 明 当 前 用 户 是 第 一 次 将 
商品 添加 到 购物 车 中 , 则 将 这 两 个 数组 元 素 的 值 设置 为 空 , 最 后 还 需要 对 购物 车 类 Cart 进行 实 
例 化 ， 以 便 继续 对 购物 车 中 商品 执行 增 、 删 、 改 操作 。 

(2) 将 商品 添加 到 购物 车 是 通过 购物 车 类 的 addCart0 方 法 实现 的 ， 在 讲解 购物 车 类 时 已 
经 介绍 过 相关 方法 的 参数 说 明 。 该 方法 所 包含 的 两 个 参数 分 别 为 购买 商品 的 ID 和 购买 商品 的 
数量 ， 将 商品 成 功 添加 到 购物 车 后 ， 还 需要 使 用 购物 车 类 的 getIdStr0 方 法 和 getNumStr0 方 法 
分 页 获取 经 购物 车 类 处 理 后 的 商品 D 串 和 商品 数量 串 ， 将 获取 结果 分 别 赋 给 $_SESSION 全 局 
数组 中 用 于 保存 商品 ID 串 和 商品 数量 串 的 数组 元 素 ， 从 而 实现 购物 车 中 的 商品 刷新 效果 。 实 
现 上 述 过 程 的 代码 如 下 : 


if($t —'add) { // 判 断 当期 操作 类 型 
$cart->addCart($_GET['bid"], 1); /调用 购物 车 类 的 addCart0 方 法 将 商品 添加 到 购物 车 中 
$_SESSION['idStr'] = $cart->getIdStr(); 1/ 获取 商品 攻 串 


$_SESSION[mumstr] = $cart->getNumStr0; /获取 商品 数量 串 
} 

(3) 在 介绍 图 书 详细 信息 页 时 ， 笔 者 已 经 介绍 过 如 何 将 与 某 本 图 书 相关 的 所 有 图 书 批量 
添加 到 购物 车 的 实现 过 程 ， 其 原理 是 首先 通过 函数 explode() 将 保存 组 合 图 书 ID 的 字符 串 用 字 
符 “@” 分 割 ， 并 将 分 割 结 果 保存 到 数组 中 ， 这 样 该 数组 的 每 个 元 素 为 各 本 组 合 图 书 的 ID 号 ， 
然后 通过 for 循环 遍历 该 数组 ， 在 遍历 过 程 中 分 别 通过 购物 车 类 的 addCart() 方 法 将 图 书 添 加 到 
购物 车 中 。 上 述 过 程 的 具体 实现 代码 如 下 : 


if($t—'gAdd") { // 判 断 操作 类 型 ， 如 果 为 gAdd 表示 组 合 添加 
$arrayGIds = explode('@', $_GET['ids"]): // 分 割 组 合 ID 并 将 结果 保存 到 数组 中 
for($i=0; $i<count($arrayGIds); $i++) { 1/ 遍历 数组 


if($arrayGIds[$i]!=" && !in_array($arrayGIds[$i], explode('@', $_SESSION['idStr])){ 
$cart->addCart($arrayGIds[$i] ,1); /将 每 本 图 书 分 别 添 加 到 购物 车 
} 


} 
$_SESSION['idStr'] = $cart->getIdStrO; // 重 新 为 商品 ID 串 的 SESSION 变量 赋值 
$_SESSION['numStr'] = $cart->getNumStr0; /重新 为 商品 数量 串 的 SESSION 变量 赋值 
| 
(4) 将 指定 的 商品 从 购物 车 中 移出 是 通过 购物 车 类 的 removeCart0 方 法 实现 的 , 该 方法 只 
包含 要 移出 商品 的 ID 参数 。 调 用 该 方法 后 ,再 分 别 为 保存 商品 ID 串 和 商品 数量 串 的 $_ SESSION 
全 局 数组 重新 赋值 ， 从 而 实现 刷新 购物 车 的 效果 。 该 过 程 的 实现 代码 如 下 : 
if($t—'remove’) { // 判 断 当 前 操 是 否 为 删除 操作 
S$cart->removeCart($ GET['rbid']): // 调 用 购物 车 类 的 removeCart0 方 法 删除 指定 的 图 书 
$_SESSION['idStr'] = $cart->getIdStrO; 
$_SESSION['numStr'] = $cart->getNumStr(): 
所 
(5) 在 实现 更 改 商品 数量 功能 时 ， 由 于 商品 数量 由 购买 者 录入 ， 所 以 在 进行 更 新 商品 数 
量 逻辑 前 ， 首 先 应 判断 购买 者 录入 的 数字 是 否 合法 ， 如 果 不 合 法 ， 则 给 出 提示 并 要 求 用 户 重 新 
填写 要 更 改 的 购买 数量 , 成 功 通过 验证 后 , 就 可 以 通过 购物 车 类 的 changNum() 方 法 实现 更 改 购 
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物 车 中 商品 的 业务 逻辑 。 该 过 程 的 实现 代码 如 下 : 
这 St 一 pdate){ 1/ 如果 $t 的 值 为 update， 则 进行 更 新 操作 ， 
$changenum = $_GET['nom']; // 获 得 要 更 新 图 书 的 数量 | 
if(!is_ numeric($changenum) || intval($changenum)<=0 || floor($changenum)!=$changenum){ 
// 判 断 数 量 是 否 为 数字 
echo "<script>alert( 购 买 图 书 的 数量 只 能 为 正 整数 ! "):</script>"; 
}else { 

/调用 购物 车 类 的 changeNum0 方 法 实现 更 改 购物 车 中 的 图 书 数量 | 
$cart->changeNum($_GET[ubid], $changenum); | 
$_SESSION['idstr] = $cart->getIdStrO; | 
$_SESSION[mumstr] = $cart->getNumStr(): | 

} | 


} 
(6) 清空 购物 车 中 商品 是 通过 购物 车 类 的 setCartNull0 方 法 实现 的 ， 调 用 该 方法 后 青 重新 
为 保存 商品 ID 串 和 商品 数量 串 的 $_SESSION 全 局 数组 赋值 即 可 。 其 实现 代码 如 下 : 


if($t 一 'clearAll) { /如 果 $t 为 clearAll， 则 表示 进行 清空 购物 车 操作 
$cart->setCartNull0: // 调 用 购物 车 类 的 setCartNull0 方 法 清空 购物 车 


$_SESSION['idStr'] = $cart->getIdStr(): 
$_SESSION['numStr'] = $cart->getNumStr(): 
} 
《7) 完成 购物 车 中 商品 的 增 、 出 、 改 操作 后 ， 还 需要 根据 保存 购物 车 商品 数量 品 的 
$_SESSION[idSt] 数 组 元 素 从 数据 库 中 查询 出 购物 车 中 各 本 图 书 的 信息 ， 并 对 每 本 图 书 进 行 价 
格 小 计 和 购物 车 中 所 有 图 书 的 价格 进行 汇总 。 具 体 实现 代码 如 下 : 


$arrayIds = explode((@', $_ SESSION['idstr]); /分 割 商品 ID 串 为 数组 
$arrayNums = explode((@',$_ SESSION[mumstr]); /分 割 商品 数量 串 为 数组 
$arrayCarInfos = array(); // 保 存 购物 车 信息 的 数组 
S$totalPrice = 0: // 购 物 车 图 书 总 价格 


for($i = 0: $1 < count($arrayIds): $i++){ 
$bookid = $arrayIds[$i]; 
if($bookid (= "){ 
S$tmpArray = array(); 
$bookinfo = $adminDB->executeSQL("select id, bookname ,oldprice, newprice from 
tb_bookinfo where id=".$bookid."", $connID): // 获 得 购物 车 每 本 图 书 的 信息 
S$tmpArray["id'] = $bookinfo[0]['id]; 
S$tmpArray['bookname'] = $bookinfo[0][bookname']; 
$tmpArray['oldprice'] = $bookinfo[0]['oldprice']; | 
S$tmpArray['newprice'] = $bookinfo[O]['newprice']; 
StmpArray['num'] = $arrayNums[$i]; | 
$tmpArray['smallTotalPrice'] = $bookinfo[0][mewprice'] * $arrayNums[$1i]; | 
S$totalPrice += $tmpArray['smallTotalPrice']; // 计 算 图 书 总 价 | 
array_push($arrayCarInfos, StmpArray): // 将 每 本 图 书信 息 保 存 到 数组 中 
9 
} | 
$smarty->assign('arrayCarInfos', $arrayCarInfos):; 
$smarty->assign('totalPrice', $totalPrice); | 
ifisset($_ SESSION[unc]) && $ SESSION[nc]'=") { /判断 用 户 是 否 登录 | 
S$smarty->assign('isLogin', T"); | 
}else{ | 
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$smarty->assign(isLogin',F): 
ifisset($_SESSION[toUrlD)){ 
unset($_SESSION['toUrI"]): 


| } 
侠 内 | $_SESSION['toUrl] = 'getbuyuserinfo.html'’; 
a | ) 


创建 的 购物 车 如 图 84 所 示 。 


您 当前 的 位 置 : 明日 网上 书店 ?》 购 关 车 


痛风 物流 思 i 


Errye7 加 mst 
E 3 mass 广 PE mtsssat 


图 书 名 入 市 场 从 (元) 。 会员 人 (元 ) 。 吉本 ) 从 将 4 计 [元 ) 所 作 
ATisead ca 项 目 开发 实例 自学 手册 cDD 。 506 45.00 FE 00 ET 
《mi 开 上 网 瑟 大 全 》 mm 2 二 四 融 而 | 六 际 
沁 购 税 | 网 雪村 ee i 从 太后 203.00 元 
退 其 提示; 


“在 本 站 风 槐 时 , 于 不 要 禁用 浏览 各 的 cook1s 支 持 ， 严 册 元 法 村 让 好 深 加 吧 风物 车 
“购买 图 书 过 各 中， 媚 议 不 要 于 用 测 扣 各 工具 磋 由 的 “ 划 违 ”和 “后 时 ”其 钮 ， 尽 旺 全 让 训 站 厅 而 内 家 中 的 邱 也 估 人 ， 记 样 可以 历 止 因 浏 贞 吉 过 丰 千 
成 鸭 物 寺中 商品 信和 销 误 


图 8.4 购物 车 
问题 182 ”如 何 制 作 柱 形 图 绘制 类 ? 


| 问题 阐述 


如 何 制作 柱 形 图 绘制 类 ? 

| 专家 解答 

| 绘制 柱 形 图 应 用 的 是 Jpgraph 类 库 中 的 BarPlot 类 , 调用 SetFillGradient0 方 法 完成 颜色 、 形 
| 状 的 设置 。 


(1) SetFillGradient0 方 法 ， 设 置 柱 形 图 的 颜色 、 形 状 。 其 语法 如 下 : 
SetFillGradient($aFromColor $aToColor, $aStyle) 

参数 说 明 : 

@ $aFromColor: 默认 颜色 。 

@ $aToColor: 转换 的 颜色 。 

@ $aStyle: 设置 的 样式 。 

(2) SetFont0 方 法 统计 图 标题 、 坐 标 轴 等 文字 样式 。 其 语法 如 下 : 
SetFont($family, [$style,] [$size]) 

参数 说 明 : 

@ $family: 指定 文字 的 字体 。 

@ $style: 指定 文字 的 样式 。 

日 $size: 指定 文字 的 大 小 ， 默 认为 10。 

(3) SetMargin() 方 法 设置 图 像 、 标 题 、 坐 标 轴 上 文字 与 边框 的 距离 。 其 语法 如 下 : 


了 。252 。 
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SetMargin($left,$right. $top, $bottom) 
参数 指定 其 与 左右 、 上 下 边框 的 距离 。 
或 者 : 
SetMargin($data) 
参数 $data 同样 指定 与 边框 的 距离 。 
应 用 示例 
在 本 示例 中 运用 Jpgraph 生成 柱 形 图 ， 对 公司 编 
程 词典 上 半年 的 销量 进行 统计 ， 其 运行 结果 如 图 8.5 
所 示 。 
(1) 创建 indexphp 文件 , 设置 网 页 的 编码 格式 ， 
并 通过 include() 语 句 导 入 Jpgraph 类 库 。 
(2) 应 用 Jpgraph 类 库 中 的 方法 完成 柱 形 图 的 创 


建 ， 其 关键 代码 如 下 : 图 8.5 编程 词典 上 半年 销量 展示 
<2php 
header ( "Content-type: text/html: charset=UTF-8" ):; // 设 置 文件 编码 格式 


include ("../src/jpgraph.php"); 

include ("../src/jpgraph_bar.php"); 
$datay=array(150,201,145,340,265,365): 
S$datax=array("Jan","Feb","Mar","Apr","May","June"): 


$graph = new Graph(400,200,"auto"); /创建 图 像 
$eraph->img->SetMargin(60,20,30,50): // 设 置 图 像 边 框 间 距 
$egraph->SetScale("textlin"); /定义 坐标 刻度 类 型 
$egraph->SetMarginColor("lightblue"); // 定 义 图 像 颜 色 
$graph->title->Set(iconv("utf-8","gb2312", 编 程 词典 上 半年 总 的 销售 量 分 析 '));// 定 义 标题 
$egraph->title->SetFont(FF_SIMSUN, FS BOLD): // 设 置 标题 字体 
$egraph->xaxis->SetFont(FF VERDANA.FS NORMAL,10); // 设 置 勺 轴 的 字体 
$egraph->yaxis->SetFont(FF VERDANA.FS NORMAL,10); /设置 立轴 的 字体 
Seraph->xaxis->SetTickLabels($datax): // 设 置 X 轴 输出 的 数据 
$egraph->xaxis->SetLabelAngle(50): /设置 输出 文字 大 小 
$bplot= new BarPlot($datay): /实例 化 图 像 创建 类 
$bplot->SetWidth(0.6): /设置 柱 形 图 的 输出 大 小 
$bplot->SetFillGradient("navy","#FFFF00",GRAD LEFT REFLECTION): 

/设置 图 像 的 类 型 和 填充 颜色 
$bplot->SetColor("white"): // 设 置 图 像 边 框 颜色 
$eraph->Add($bplot): // 添 加 数据 
$graph->Stroke0: /生成 图 像 
?> 


问题 183 如何 制 作 折线 图 绘制 类 ? 


问题 阐述 
如 何 制作 折线 图 绘制 类 ? 


“253。 
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绘制 折线 图 应 用 的 是 卫 graph 类 库 中 的 LinePlot 类 。 输 出 数据 应 用 的 是 value0 对 象 中 的 

Show() 方 法 ， 折 线 图 中 标记 的 添加 应 用 的 是 mark 对 象 的 SetType0 方 法 。 
(1) 使 用 LinePlot 对 象 绘制 曲线 。 
通过 Jpgraph 类 库 中 的 LinePlot 类 创建 曲线 。 该 类 的 语法 如 下 : 
$linePlot= new LinePlot($data) // 创 建 折线 图 

参数 说 明 : 

$data: 数值 型 数组 ， 指 定 统计 数据 。 
(2) 输出 数据 应 用 的 是 value0 对 象 中 的 Show 方法， 数据 的 格式 化 应 用 的 是 SetFormat() 


SetFormat($aFormat, $aNegFormat) 
参数 说 明 : 
@ $aFormat: 设置 格式 化 的 样式 。 
@ $aNegFormat: 设置 negative 的 值 。 
(3) 折线 图 中 标记 的 添加 应 用 的 是 图 像 中 的 mark 对 象 的 SetType0 方 法 , 其 参数 值 设置 标 


| 记 的 类 型 。 其 语法 如 下 ; 


SetType($aType, $aFileName, $aScale) 
参数 说 明 : 
@ $aType: 设置 标记 的 类 型 。 
@ $aFileName: 插入 图 像 的 名 称 或 国家 的 名 称 。 
人 @@ $aScale: 设置 插入 标记 的 大 小 。 
下 面 列举 了 一 些 SetType(0 方 法 中 第 一 个 参数 可 选 的 类 型 : MARK SQUARE、MARK_ 


| UTRIANGLE、 MARK DTRIANGLE MARK DIAMOND、MARK _ CIRCLE。 


应 用 示例 

在 本 示例 中 运用 Jpgraph 生成 折线 图 ， 对 公司 2010 年 的 销售 额 进行 分 析 ， 其 运行 结果 如 
图 8.6 所 示 。 
eT wr 


i | Wd 


图 8.6 ”2010 年 公司 的 销售 额 


折线 图 分 析 2010 年 公司 销售 额 走势 的 关键 步骤 如 下 : 
。254 。 
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(1) 创建 index php 文件 ， 设 置 网 页 的 编码 格式 ， 并 通过 include0 语 句 导 入 Jpgraph 类 库 。 
(2) 应 用 Jpgraph 类 库 中 的 方法 创建 一 个 折线 图 ， 对 2010 年 公司 的 销售 额 进行 分 析 。 其 | 
代码 如 下 : | 
<2php 
header ( "Content-type: text/html:; charset=UTF-8" ); /设置 文件 编码 格式 
include ("../src/jpgraph.php"); 
include ("../src/jpgraph_line.php"); 
include ("../src/jpgraph_scatter.php"); ! 


问题 184 ”如 何 制 作 人 饼 形 图 绘制 类 ? 


$datayl = array(40,26,15,35,26,15,44,26,15,27,40,15); | 
$graph = new Graph(650.375); /定义 图 像 大 小 | 
$eraph->SetMarginColor('white'); 1/ 背景 颜色 | 
$eraph->SetScale("textlin"); // 定 义 刻度 值 类 型 | 
Seraph->SetFrame(false); | 
$egraph->SetMargin(30,5,35,20); /设置 边 距 | 
$graph->tabtitle->Set(iconv("utf-8","gb2312",'2010 年 销售 额 分 析 ) ); /输出 标题 | 
$eraph->tabtitle->SetFont(FF_SIMSUN, FS_BOLD.12): /设置 标题 字体 | 
$egraph->tabtitle->SetColor('darkred',#E1E1FF"): /设置 标题 颜色 | 
$egraph->xgrid->Show(); /设置 阴影 ! 
$graph->xaxis->SetTickLabels($gDateLocale->GetShortMonth0); /定义 和 X 轴 上 的 数据 | 
$pl = new LinePlot($datay]); // 创 建 图 像 | 
S$p1->SetColor("navy"); // 设 置 图 像 颜 色 | 
$p1->mark->SetType(MARK IMG,'saab 95.jpg.0.5); /设置 标签 的 样式 ， 使 用 图 片 ， 
$p1->value->SetFormat('%d '.iconv("utf-8","gb2312", 万 元 "); /格式 化 数据 | 
$p1->value->Show0; /输出 阴影 | 
$pl1->value->SetColor('darkred'); // 定 义 颜 色 | 
$p1->value->SetFont(FF_SIMSUN, FS BOLD,10): /设置 字体 | 
$pl->value->SetMargin(14): // 设 置 位 置 、 字 体 大 小 | 
$p1->SetCenter0); | 
$eraph->Add($p1): /添加 数据 | 
$egraph->Stroke(): // 生 成 图 像 | 
?> | 


问题 阐述 
如 何 制作 饼 形 图 绘制 类 ? 
专家 解答 | 
绘制 3D 饼 形 图 应 用 的 是 PiePlot3D 类 ， 继 承 PiePlot 类 。 | 
通过 PiePlot3D 类 中 的 Explode() 方 法 对 饼 形 图 进行 分 割 。Explode0 方 法 语法 如 下 : | 
Explode($aExplodeArr) | 


将 饼 形 图 分 割 成 多 个 模块 ， 参 数 $SaExplodeArr 是 一 个 数组 ， 根 据 数组 元 素 个 数 定义 分 割 成 
几 块 ， 根 据 数组 元 素 值 定义 分 割 后 模块 与 原 图 像 的 距离 。 
通过 饼 形 图 对 象 调用 SetTheme() 方 法 设置 饼 形 图 的 颜色 。SetTheme( 方 法 的 语法 如 下 : 


"255。 
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SetTheme($aTheme) 


参数 $aTheme 指定 主题 的 名 称 。 其 支持 的 主题 包括 :“earth”、“pastel”、“sand”、“water”。 
通过 饼 形 图 对 象 调用 SetTheme() 方 法 设置 饼 形 图 的 颜色 。SetTheme() 方 法 的 语法 如 下 : 
PiePlot :: 


be | SetTheme($aTheme) 


| PiePlot :: 


参数 $aTheme 指定 主题 的 名 称 。 其 支持 的 主题 包括 :“earth”、“pastel”、“sand”、“water”。 


应 用 示例 | 
在 本 示例 中 运用 Jpgraph 生成 饼 形 图 ， 对 公司 一 sx 
2010 年 图 书 销量 进行 分 析 , 其 运行 结果 如 图 87 所 示 。 | 将 ”| 


饼 形 图 分 析 2010 年 图 书 销量 的 关键 步骤 如 下 : 
(1) 创建 index.php 文件 ， 设 置 网 页 的 编码 格 
式 ， 通 过 include() 语 句 导 入 Jpgraph 类 库 。 


(2) 应 用 Jpgraph 类 库 中 的 方法 创建 饼 形 图 ， 图 8.7 2010 年 图 书 销量 分 析 
对 2010 年 图 书 销量 进行 分 析 。 其 代码 如 下 : 
<2p| 
header ( "Content-type: text/html:; charset=UTF-8" ): // 设 置 文件 编码 格式 


include ("../src/jpgraph.php"); 
include ("../src/jpgraph_pie.php"); 
include ("../src/jpgraph_pie3d.php"); 


$data = array(20,27,45,75,90,30): 
$graph = new PieGraph(500,245,"auto"): // 创 建 图 像 
$graph->SetShadow0; /创建 图 像 阴影 
$egraph->tabtitle->Set(iconv("utf-8","gb2312",'2010 年 图 书 销售 分 析 ) );，”// 输 出 标题 
$graph->tabtitle->SetFont(FF_SIMSUN,FS_BOLD,14); /设置 标题 字体 
$eraph->title->SetColor("darkblue"): // 定 义 标题 颜色 
$graph->legend->Pos(0.1.0.2): /控制 注释 文字 的 位 置 
$pl = new PiePlot3d($data): /创建 图 像 
$p1->SetTheme("sand"); // 控 制图 像 颜色 
$p1->SetCenter(0.4): /设置 图 像 位 置 
$p1->SetSize(0.4): /设置 图 像 大 小 

| $p1->SetHeight(20); /设置 饼 形 图 高 度 

| $p1->SetAngle(45): /设置 图 像 倾斜 角度 

| $p1->Explode(array(5,40,10,30,20)): // 控 制 饼 形 图 的 分 割 

| $p1->value->SetFont(FF_ARIAL.FS NORMAL.10): /设置 字体 

| $p1->SetLegends(array("ASPNET","C#","JAVA"."PHP","VB","VC")): 

| $eraph->Add($p1): /添加 数据 

| $graph->Stroke0: // 生 成 图 像 

! ?> 


问题 185 ”如 何 封装 验证 码 类 ? 


”问题 阐述 
| 验证 码 功能 在 注册 、 登 录 、 发 布 信息 和 评论 中 经 常会 用 到 ， 为 了 减少 代码 的 元 余 以 及 提高 
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代码 的 重用 率 ， 可 以 将 该 功能 封装 到 类 中 ， 在 需要 时 直接 调用 类 中 方法 即 可 实现 验证 码 功能 。 
那么 如 何 封 装 验 证 码 类 呢 ? 
专家 解答 
1. 类 的 封装 
首先 ， 定 义 验 证 码 类 的 名 称 是 ValidateCode。 该 类 的 语法 如 下 : 
/六 六 六 六 六 六 六 六 六 闵 闵 素 六 水 六 六 六 六 冰 闵 六 这 六 水 闵 六 这 六 闲 水 六 六 六 六 六 六 六 率 六 六 闵 闲 率 六 冰 
* @ 说 明 : 验证 码 类 
*@ 作者 : 明日 科技 | 
*@ 学 习 社 区 : www.mrbccd.com | 
* @ E-mail:mingrisoft@mingrisoft.com 
六 闵 率 六 六 闵 闵 洲 六 六 六 水 六 六 水 六 水 率 六 水 六 来 闵 六 六 六 六 率 闵 冰冰 六 六 率 闵 水 六 闵 率 冰冰 冰冰 来 / 
class ValidateCode{ 
// 宽 度 
private $_width; 
/长 度 
Private $_height; 
// 验 证 码 
private $_codeStr; 
/字体 类 型 
private $_fontType; 
/图 像 名 柄 
private $_img; 
ai 
* 构造 方法 
* (@param string $width 
* (@param string $height 
* (Qparam string $codeStr 
* (Oparam int $fontType 
$y 
public function __construct ($width, $height, $codeStr = '0000', $fontType = 0){ 
S$this->_width = $width; 
$this-> height = $height: 
S$this->_codeStr = substr($codeStr 0, 4): 
$this-> fontType = $fontType; 


/时 | 


* 获取 颜色 
来 


* @param 颜色 范围 最 小 值 $x 
* @param 颜色 范围 最 大 值 $y | 
* (return imagecolorallocate() 
由 | 
private function _getColor ($x, $y) { | 
S$r = mt _ rand($x, $y):; | 
Sg 一 mt rand(Sx, $y); | 
$b = mt rand($x, $y); 
Teturn imagecolorallocate($this-> img, Sr $g, $b); 


2 
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wed 
private function _init O{ 
S$this-> img = imagecreate($this-> width, $this-> height); 
} 


/本 水 


* 创建 验证 码 


站 
Private function _ build | { 
imagefill($this-> img, 0, 0, $this->_getColor(150, 250)); 
imagerectangle($this-> img, 0, 0, $this-> width - 1, Sthis-> height - 1, $this-> getColor(50, 150)); 
if($this-> fontType — 0) { 
SfontFileName = 'ARIALBI.TTF'; 
} else { 
SfontFileName ='ARIALN.TTF': 


for ($i= 0:; $1< strlen($this-> codeStr): $i++) { 
| imagettftext($this-> img, mt rand(12, 24), 0, ($this-> width) / 4 * $i, mt rand(20, 
| Sthis-> height - 5), $this-> getColor(10, 180), 'font/. $fontFileName, substr($this->_codeStr, $i, 1)); 
| 


for ($1= 0; $i1< 15; $i1++) { 
| imageline($this-> img, mt rand(0, $this-> width)， mt rand(0, $this-> height), 
| mt_rand(0, $this-> width), mt_rand(0, $this->_ height), $this-> _getColor(110, 210)); 
| ) 

/** 
* 显示 图 像 
Eh 
public function show Of 

header(content-type:image/png ); 

$this-> initO: 

$this->_ build0; 

imagepng($this-> img): 
| 
| } 
2. 应 用 示例 
在 本 示例 中 编写 一 个 用 户 登录 功能 , 调用 验证 码 类 生成 验证 码 , 通过 jQuery 在 客户 端 完 成 
| 对 用 户 登录 信息 的 验证 。 其 具体 步骤 如 下 : 
| (1) 封装 ValidateCode 验证 码 类 ， 存 储 于 ValidateCode php 文件 中 。 
| (2) 创建 index.php 文件 ， 设 计 用 户 登录 界面 ， 添 加 用 户 登 录 的 表单 元 素 。 
| (3) 创建 yzm.php 文件 ， 生 成 验证 码 ， 并 且 在 index.php 文件 中 以 图 像 形式 输出 。 其 关键 
| 代码 如 下 : 


<?php 
session start|; // 初 始 化 SESSION 变量 
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include("ValidateCode .php"); // 载 入 验证 码 生 成 类 

$codeStr =""; // 定 义 验证 码 变量 

ScodeArray = array('0' ,1 ,2 ,3 ,4 ,5,16' ,7 BC ID E', F','G' 于 了， 
KL , M,N ,0', P,Q , R's, TT, ,VW TD 

for ($i1= 0; $1<4; $i1++) { 


ScodeStr = $codeArray[mt rand(0, 35)]; /生成 随机 验证 码 数据 
$_SESSION['Vcode']=$codeStr: // 将 验证 码 数据 赋 给 SESSION 变量 
$validateCode = new ValidateCode(80, 30, $codeStr, 0): // 实 例 化 验证 码 类 
$validateCode->show(); // 生 成 验证 码 
?> 

(4) 创建 JavaScript 脚本 ， 编 写 changeVcode() 方 法 ， 实 现 验证 码 的 切换 。 其 代码 如 下 : 
function changeVcodeO{ 

S$("#vcodeImg").attr("sre", "yzm.php?rand="+Math.random()+"")"); 

) 


(5) 编写 JavaScript 脚本 ， 通 过 jQuery 技术 完成 对 表单 中 提交 数据 的 验证 。 其 中 要 特别 
注意 的 是 调用 的 check_code.php 文件 ， 在 该 文件 中 对 用 户 提交 的 验证 码 进行 验证 。JavaScript 
脚本 的 代码 如 下 : 


// 表 单 验 证 
function chkInputLogin(flag){ 
$.get("check_code.php?veode="+encodeURI($("#vcode").valO), null, fonction(chkVcodeRetumData){ 
var kl=false:; 
Var k2=false: 
Var k3=false; 
/用 户 昵称 
if(flag — 1 | flag — -1){ 
if($.trim($("#netname").val))=—""){ 
S$("#chkNetname").html(" 请 输入 登录 昵称 "); 
S$("#netname").css("border", "1px solid #FF0000"); 
S$("#chkNetname").css("display", "block"): 
Jelse{ 
$("#chkNetname").html(""); 
S$("#netname").css("border", "1px solid #777777"); 
$("#chkNetname").css("display", "none"); 
kl=true; 
B 


} 
// 登 录 密码 
if(flag —2|flag—-D{ 
if($.trim($("#password").val0)—""){ 
S$("#chkPassword").html(" 请 输入 登录 密码 "); 
S$("#password").css("border", "1px solid #EfF0000"): 
S$("#chkPassword").css("display", "block"): 
Jelse{ 
S$("#chkPassword").html(""); 
S$("#password").css("border", "1px solid #777777"): 
S$("#chkPassword").css("display", "none"): 
k2=true: 


“2 


~ 可 (本 > 交 需 知 造 的 300 个 癌症 


有 
if(flag —3 | flag —-D{ 
if($.trim($("#veode").val0)—""){ 


) 
| /验证 码 


$("#chkVcode").html(” ”请 输入 验证 码 "); 

$("#vcode").css("border", "1px solid #FF0000"); 

$("#chkVeode").css("display", "block"); 

}else 这 chkVcodeRetumData 一 "N"){ 
$("#chkVcode").html(” ”验证 码 输入 有 误 "); 
$("#vcode").css("border", "1px solid #FF0000"); 
$("#chkVeode").css("display", "block"): 

Jelse{ 

S$("#chkVeode").html("™"); 
$("#vcode").css("border", "1px solid #777777"): 
S$("#chkVeode").css("display", "none"); 
k3=true; 

}; 


» 

// 提 交 表 单 

if(flag — -1 && kl && k2 && 3 ){ 
S$("#form login").submitO: 

} 


从 


记忆 
} 
| 运行 结果 如 图 8.8 所 示 。 


图 8.8 用 户 登 录 时 的 验证 码 
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于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 于 至 


什么 是 PDO? 

如 何 安 装 PDO? 

如 何 通 过 PDO 访问 数据 库 ? 

如 何 通 过 PDO 读 取 数据 表 中 的 数据 ? 

如 何 解 决 SQL 查询 中 的 错误 ? 

如 何 向 数据 库 中 添加 或 修改 数据 ? 

如 何 保护 Web 站 点 免 受 SQL 注入 攻击 ? 
如 何 获得 受 影响 的 行 数 ? 

如 何 进行 事务 处 理 ? 

如 何 利用 PDO 使 用 存储 过 程 ? 

如 何 利 用 PDO 连接 Oracle 11g? 

如 何 封 装 PDO 数据 库 连 接 类 ? 

如 何 封装 PDO 数据 库 操 作 类 ? 

如 何 封装 PDO 分 页 类 ? 

如 何 封装 ADODB 连接 数据 库 类 ? 

ADODB 操作 结果 集 的 方法 有 哪些 ? 
ADODB 中 如 何 控制 结果 集 的 存 取 方法 ? 
如 何 一 次 连接 两 个 数据 库 ? 

如 何在 ADODB 中 生成 HTML 表格 ? 

如 何在 ADODB 中 生成 下 拉 列 表 框 ? 

一 个 小 巧 的 分 页 函数 一 一 ADODB_Pager0 
如 何 通过 事务 处 理 机 制 保证 数据 库 操作 的 完整 性 ? 
ADODB 如 何 输出 系统 中 执行 的 SQL 语句 ? 
ADODB 如 何 应 用 ErrorMsg 返回 错误 信息 ? 
ADODB 如 何 返回 所 有 错误 信息 ? 

如 何 将 错误 信息 存储 于 日 志文 件 中 ? 

如 何 封装 ADODB 操作 数据 库 类 ? 

如 何 封 装 ADODB 分 页 类 ? 


需 
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问题 186 ”什么 是 PDO? 


问题 阐述 
什么 是 PDO? PDO 有 什么 特点 ? 


专家 解答 


PDO 是 PHP Date Object (PHP 数据 对 象 ) 的 简称 ， 它 是 与 PHP 5.1 版 本 一 起 发 行 的 ， 目 前 


| 支持 的 数据 库 包括 Firebird、FreeTDS、Interbase、MySQL、MSSQL Server、ODBC、Oracle、 
| Postgre SQL、SQLite 和 Sybase。 有 了 PDO， 用 户 不 必 再 使 用 mysql *、oci_* 及 mssql * 函 数 ， 
| 也 不 必 再 为 它们 封装 数据 库 操作 类 ， 只 需要 使 用 PDO 接口 中 的 方法 就 可 以 对 数据 库 进行 操作 。 
| 在 选择 不 同 的 数据 库 时 ， 只 需 修改 PDO 的 DSN (数据 源 名 称 〉 即 可 。 


在 PHP 6 中 将 默认 使 用 PDO 连接 数据 库 ， 所 有 非 PDO 扩展 将 会 在 PHP 6 中 被 移 除 。 该 扩 


| 展 提供 PHP 内 置 类 PDO 来 对 数据 库 进行 访问 ， 不 同 数据 库 使 用 相同 的 方法 名 解决 数据 库 连接 
| 不 统一 的 问题 。 


PDO 是 一 个 “数据 库 访 问 抽象 屋 "， 其 作用 是 统一 各 种 数据 库 的 访问 接口 ， 与 MySQL 和 


| MSSQL 函数 库 相 比 ，PDO 让 跨 数据 库 的 使 用 更 具有 亲和力 ; 与 ADODB 和 MDB2 相 比 , PDO 


| 更 高 效 。 


PDO 将 通过 一 种 轻型 、 清 晰 、 方 便 的 函数 ， 统 一 各 种 不 同 RDBMS 库 的 共有 特性 ， 实 现 


| PHP 脚本 最 大 程度 的 抽象 性 和 兼容 性 。 


PDO 吸取 现 有 数据 库 扩展 成 功 和 失败 的 经 验 教训 , 利用 PHP 5 的 最 新 特性 , 可 以 轻松 地 与 


| 各 种 数据 库 进 行 交 互 。 


PDO 扩展 是 模块 化 的 , 使 其 能 够 在 运行 时 为 用 户 的 数据 库 后 端 加 载 驱 动 程序 ,而 不 必 重 新 


| 编译 或 安装 整个 PHP 程序 .例如 ,PDO_MySQL 扩展 会 葵 代 PDO 扩展 实现 MySQL 数据 库 API。 
| 另外 ， 还 有 一 些 用 于 Oracle、PostgreSQL、ODBC 和 Firebird 的 驱动 程序 ， 更 多 的 驱动 程序 尚 
| 在 开发 。 


问题 187 如何 安装 PDO? 


问题 阐述 


| 对 象 特性 的 支持 ， 因 此 其 无 法 在 PHP 5.0 之 前 的 版 本 中 使 用 。 


如 何 安装 PDO? 


专家 解答 


PDO 是 与 PHP 5.1 一 起 发 行 的 ， 默 认 包含 在 PHP 5.1 中 。 由 于 PDO 需要 PHP 5 核心 面向 


在 默认 情况 下 , PDO 在 PHP 5.2 中 为 开启 状态 , 但 是 要 启用 对 某 个 数据 库 驱 动 程序 的 支持 ， 


， 仍 需要 进行 相应 的 配置 操作 。 


在 Linux 环境 下 ， 要 使 用 MySQL 数据库， 可 以 在 configure 命令 中 添加 如 下 选项 : 
--with-pdo-mysql=/path/to/mysql/installation 
*262。 


在 Windows 环境 下 ，PDO 在 php.ini 文件 中 进行 配置 ， 


如 图 9.1 所 示 。 


要 启用 PDO， 首 先 必须 加 载 “extension=php_pdo.dll”， 
如 果 要 想 其 支持 某 个 具体 的 数据 库 , 那么 还 要 加 载 对 应 的 数 
据 库 选项 。 例 如 ， 要 支持 MySQL 数据 库 ， 则 需要 加 载 


“extension=php_pdo_ mysqldll” 选 项 。 


专家 点 评 

在 完成 数据 库 的 加 载 后 还 要 保存 php.ini 文件 ， 并 且 重 新 启动 Apache 服务 器 ， 修 改 才 能 够 
生效 。 

问题 188 ”如 何 通过 PDO 访问 数据 库 ? 

问题 阐述 

如 何 通 过 PDO 访问 数据 库 ? 
专家 解答 

在 PDO 中 , 要 建立 与 数据 库 的 连接 需要 实例 化 PDO 的 构造 函数 。 PDO 构造 函数 的 语法 如 下 : 


_construct(string $dsn[,string $usernamel[,string $password[,array $driver_options]]]) 


参数 说 明 : 


@ dsn: 数据 源 名 ， 包 括 主机 名 端口 号 和 数据 库 名 称 。 


@ username: 连接 数据 库 的 用 户 名 。 
@@ password: 连接 数据 库 的 密码 。 
@ driver_ options: 连接 数据 库 的 其 他 选项 。 


1. 连接 MySQL 数据 库 


sextensio! 
lextension=| 


Php_pdo_fireb: 
hp_Pao_nssq. 
dt a1 


ppdo_o 
nm=php_pdo_pgsq: 
php_pdo_sqlite.dll 


图 9.1 Windows 环境 下 配置 PDO 


下 面 的 代码 展现 了 如 何 连 接 到 一 个 本 地 主机 上 的 MySQL 数据 库 。 


<2php 
header("Content-Type:text/html:;charset=utf-8"); 
$dbms="mysql'; 
$dbName='db_database'; 
$user='root’; 
Spwd="111'; 
$host='"localhost'; 
$dsn="$dbms:host=$host:dbname=$dbName"; 
try 
S$pdo=new PDO($dsn,$user, $pwd); 
echo "PDO 连接 MySQL 成 功 "; 
} catch (Exception $e) { 
echo $e->getMessage()."<br>"; 
} 


> 


“263 。 


/设置 页 面 的 编码 格式 
/数据 库 类 型 

/使 用 的 数据 库 名 称 

// 使 用 的 数据 库 用 户 名 
/使 用 的 数据 库 密码 
/使 用 的 主机 名 称 


/捕获 异常 
/实例 化 对 象 


ry 人 Pp 六 须知 道 的 300 个 问题 
2. 连接 SQL Server 数据 库 
使 用 以 下 代码 连接 到 本 地 主机 上 的 SQL Server 数据 库 。 


<2php 
header("Content-Type:text/html:charset=utf-8") // 设 置 页 面 的 编码 风格 
$host='PC-201006101638'; /设置 主机 名 称 
$user='sa'; // 设 置 用 户 名 
S$pwd="; // 设 置 密码 
$dbName='db_database'; // 设 置 所 要 连接 的 数据 库 
$dbms='mssql'; 
$dsn="mssql:host=$host;dbname=$dbName"; 
try{ // 利 用 try...catch 捕捉 异常 并 实现 连接 
$pdo = new PDO($dsn, $user,$pwd); 
echo "连接 SQL SERVER 成 功 "; 
} catch (Exception $e) { 


die("ERROR!".$e->getMessage()."<br>"); 
} 
?> 


3. 连接 Oracle 数据 库 
使 用 以 下 代码 连接 到 本 地 主机 上 的 Oracle 数据 库 。 


<2php 

$dbms='oci' /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 
库 ， 只 要 修改 它 即 可 ， 不 用 记 住 那么 多 的 函数 

$dbName='192.168.1.59:1521/oralcles': // 使 用 的 数据 库 

$user='system'; // 数 据 库 连接 用 户 名 

$pass="mrsoft'; // 对 应 的 密码 

$dsn="$dbms:dbname=$dbName": 

ty{ 

$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连接 

对 象 $pdo 


echo "连接 成 功 <br/>"; 

$pdo = mull; 
} catch (PDOException $e) { 

die ("Error!: " . $e->getMessage() . "<br/>"); 
) 


?> 


说 明 : 

通过 PDO 连 接 Oracle 数 据 库 ,首先 要 在 本 机 上 安装 Oracle 数 据 库 的 客户 端 .然后 ,去 掉 php.ini 
文件 中 extension=php_pdo.dll 和 extension=php_pdo_oci.dll 前 面 的 分 号 ， 加 载 PDO 模块 。 最 后 ， 
将 Oracle 客户 端的 oraociei10.dll、oci.dll 和 orannzsbb10.dll 复制 到 Apache 的 bin 文件 夹 下 。 


专家 点 评 
在 使 用 不 同 的 数据 库 时 ， 必 须 明 确 数据 库 服 务 器 是 完全 独立 于 PHP 的 实体 ， 即 数据 库 服 
务 器 可 能 与 Web 服务 器 不 在 同一 台 计 算 机 上 , 此 时 要 通过 PDO 连接 数据 库 时 , 就 需要 修改 DSN 
中 的 主机 名 称 。 
DSN 是 Data Source Name (数据 源 名 称 ) 的 首 字母 缩写 ， 其 提供 连接 数据 库 需 要 的 信息 。 
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PDO 的 DSN 包括 3 部 分 : PDO 驱动 名 称 (如 mysql、sqlite 或 pgsql)、 冒 号 和 驱动 特定 的 语法 。 
每 种 数据 库 都 有 其 特定 的 驱动 语法 。 | 

由 于 数据 库 服务 器 只 在 特定 的 端口 上 监听 连接 请 求 。 每 种 数据 库 服务 器 具有 一 个 默认 的 端 | 
口号 (MySQL 是 3306)， 但 是 数据 库 管理 员 可 以 对 端口 号 进行 修改 ， 因 此 有 可 能 PHP 找 不 到 | 会 内 
数据 库 的 端口 ， 此 时 就 可 以 在 DSN 中 包含 端口 号 。 ' 窒 一 

另外 ， 由 于 一 个 数据 库 服务 器 中 可 能 拥有 多 个 数据 库 ， 所 以 在 通过 DSN 连接 数据 库 时 ， 
通常 都 包括 数据 库 名 称 ， 这 样 可 以 确保 连接 的 是 用 户 想 要 的 数据 库 ， 而 不 是 其 他 人 的 数据 库 。 


问题 189 如何 通过 PDO 读 取 数 据 表 中 的 数据 ? 


问题 阐述 
既然 通过 PDO 可 以 实现 与 数据 库 的 连接 ， 那 么 如 何 通过 PDO 读 取 数 据 表 中 的 数据 呢 ? 
专家 解答 
PDO 为 用 户 提 供 了 很 多 与 数据 库 交 互 的 方法 。 其 中 最 常用 的 是 exec()、query()、prepare() | 
和 execute() 方 法 。 | 
(1) exec() 方 法 返回 执行 后 受 影响 的 行 数 ， 其 语法 如 下 : 
int PDO::exec ( string statement ) 
参数 statement 是 要 执行 的 SQL 语句 。 该 方法 返回 执行 查询 时 受 影响 的 行 数 ， 通 常用 于 
INSERT、DELETE 和 UPDATE 语句 中 。 
(2) query() 方 法 通常 用 于 返回 执行 查询 后 的 结果 集 。 其 语法 如 下 : 
PDOStatement PDO::query ( string statement ) 
参数 statement 是 要 执行 的 SQL 语句 ， 它 返回 的 是 一 个 PDOStatement 对 象 。 | 
(3) 预 处 理 语句 包含 prepare0 和 execute() 两 个 方法 。 首先, 通过 prepare() 方 法 做 查询 的 准 ， 
备 工作 ,然后 ,通过 execute() 方 法 执行 查询 ， 并 且 还 可 以 通过 bindParam() 方 法 来 绑 定 参 数 提供 | 
给 execute() 方 法 。 其 语法 如 下 : | 


PDOStatement PDO::prepare ( string statement [, array driver_options] ) 
bool PDOStatement::execute ( [array input_ parameters] ) 


1. 使 用 query() 方 法 执行 查询 语句 
通过 query0 方 法 执行 SQL 语句 ， 循 环 输出 查询 结果 。 其 关键 代码 如 下 : 
<?php 


$dbms='mysql: /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 修改 它 即 可 ， 不 用 | 
记 住 那么 多 的 函数 | 


T 


$host='localhost': /| 数据 库 主机 名 
$dbName='db_database'; // 使 用 的 数据 库 
$user=root'; // 数 据 库 连 接 用 户 名 
S$pass="111'; // 对 应 的 密码 
$dsn="$dbms:host=$host:dbname=$dbName": 

try{ 


$pdo =new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ,就 是 创建 了 数据 库 连 接 对 象 $pdo | 
5 Ce 


人 Pp 六 须知 道 的 300 个 问题 


} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage(0 . "<br/>"); 


I CD 
| &% 
| $query="select * from tb pdo_mysql"; /定义 SQL 语句 
| $result=$pdo->query($query); // 执 行 查询 语句 ， 并 返回 结果 集 
! foreach($result as $items){ 
| Wy 
”< <t> 
食 生 | | <td height="22" align="center" valign="middle"><?php echo $items['id'"]:?></td> 
| <td align="center" valign="middle"><?php echo $items['pdo_type'];?></td> 
<td align="center" valign="middle"><?php echo Sitems['database_name']:?></td> 
| <td align="center" valign="middle"><?php echo $items["dates']:2></td> 
| </tr> 
| <2php 
| } 
| } 
?> 
运行 结果 如 图 9.2 所 示 。 0 
2. 使 用 prepare() 和 execute() 方 法 执行 查询 语句 Ea 
使 用 prepare0 和 execute() 方 法 通常 被 认为 是 处 加 PDo 囊 拓 和 四 间 


理 数据 库 查 询 的 最 理想 的 方法 。 在 本 示例 中 ， 首 先 
以 SQL 语句 为 参数 调用 PDO->prepare() 方 法 ， 返 回 
-个 PDOStatement 对 象 ， 然 后 调用 execute() 方 法 执 


rr 2010-09-18 101917 
2010-09-18 101947 
ss 2010-10-10 000000 
2010-10-09 105930 


2010-10-16 1059-41 
Errerr | 
2010-10-10 101010 


行 查询 语句 , 返回 查询 结果 集 , 最 后 通过 while 循环 pe 

重复 调用 PDOStatement-> fetch() 方 法 输出 查询 结果 

集 。 其 关键 代码 如 下 : 图 9.2 使 用 PDO 的 query0 方 法 产生 的 输出 
<2php 
$dbms='mysql'; /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 

要 修改 它 即 可 ， 不 用 记 住 那么 多 的 函数 
$host='localhost'; /数据库 主 机 名 
| $dbName='db_database'; // 使 用 的 数据 库 

$user='root'; // 数 据 库 连接 用 户 名 
$pass='111' // 对 应 的 密码 


$dsn="$dbms:host=$host;:dbname=$dbName"; 


try{ 
$pdo = new PDO($dsn, $user, $pass); 。“// 初 始 化 一 个 PDO 对 象 ,就 是 创建 了 数据 库 连接 对 象 $pdo 


| $query="select * from tb_pdo_mysql": // 定 义 SQL 语句 

| $result=$pdo->prepare($query); // 准 备查 询 语 句 

| $result->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 

| while($res=$result->fetch(PDO::FETCH_ASSOC)){ // 循 环 输出 查询 结果 集 ， 并 且 设 置 结果 
| 集 为 关联 索引 

| ?> 

| 3 


| <td height="22" align="center" valign="middle"><?php echo $res[id]:?></td> 
| <td align="center" valign="middle"><?php echo $res[pdo_type']:?></td> 

| <td align="center" valign="middle"><?php echo S$res['database_name'];?></td> 
! <td align="center" valign="middle"><?php echo S$res['dates']:?></td> 
| </tr> 

| <?php 

】 
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} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 
b 
全 把 
专家 点 评 二 谢 


在 选择 SQL 语句 的 执行 方法 时 ， 如 果 只 执行 一 次 查询 ， 那 么 PDO->query0 是 较 好 的 选择 。 
虽然 它 无 法 自动 转 义 发 送 给 它 的 任何 数据 , 但 它 在 遍历 SELECT 语句 的 结果 集 方面 的 确 非常 方 | 
便 ,然而 在 使 用 这 个 方法 时 应 十 分 小 心 。 如 果 没 有 在 结果 集中 获取 到 所 有 数据 ， 那 么 下 次 调用 | 
PDO->query0 方 法 时 可 能 会 失败 。 | 

如 果 要 多 次 使 用 SQL 语句 , 那么 最 好 的 方法 是 使 用 prepare0 和 execute() 方 法 一 一 这 是 优先 | 
选择 的 方案 。 使 用 prepare() 和 execute0 方 法 与 使 用 query0 方 法 比较 起 来 有 很 多 优势 。 首 先 ， 它 | 
通过 自动 转 义 任何 提供 给 它 的 参数 帮助 防止 SQL 注入 攻击 。 其 次 ， 多 次 使 用 例如， 执行 多 | 
次 数据 库 插入 或 更 新 ) 的 预 编译 语句 占用 较 少 的 资源 , 运行 速度 要 比重 复 调 用 query() 方 法 快 很 | 
多 。 另 外 ， 当 用 prepare0 和 execute() 方 法 时 ， 还 可 以 选择 很 多 种 返回 数据 格式 。 例 如 ， 在 示例 | 
中 使 用 fetch() 方 法 时 ， 定 义 结果 集 采 用 关联 索引 (PDO::FETCH ASSOC) 的 形式 。 | 


问题 190 ”如何 解决 SQL 查询 中 的 错误 ? 


问题 阐述 | 
普 误 不 可 避免 ,它们 困扰 所 有 的 人 ,经常 由 于 无 法 控制 的 情况 一 一 数据 添加 失败 、 数 据 更 新 | 
失败 或 者 不 能 正确 删除 数据 等 造成 。 那么 , 该 如 何 捕获 SQL 语句 中 的 错误 , 并 及 时 加 以 解决 呢 ? | 
专家 解答 
PDO 为 捕获 错误 提供 了 如 下 3 种 解决 方案 。 
加 ”使 用 默认 模式 一 PDO::ERRMODE_SILENT 
在 默认 模式 中 设置 PDOStatement 对 象 的 errorCode 属性 ， 但 不 进行 其 他 任何 操作 。 
加 ”使 用 敬告 模式 一 PDO::-ERRMODE_WARNING 
警告 模式 会 产生 一 个 PHP 警告 ， 并 设置 errorCode 属性 。 如 果 设 置 的 是 警告 模式 ， 那 么 除 | 
非 明 确 地 检查 错误 代码 ， 否 则 程序 将 继续 按照 其 方式 运行 。 | 
使 用 异常 模式 一 一 PDO::ERRMODE _EXCEPTION | 
异常 模式 会 创建 一 个 PDOException， 并 设置 erorCode 属性 。 它 可 以 将 执行 代码 封装 到 一 | 
个 try{.…}catch{.…} 语 句 块 中 。 未 捕获 的 异常 将 会 导致 脚本 中 断 ， 并 显示 堆栈 跟踪 让 用 户 了 解 | 
是 哪里 出 现 的 问题 。 | 
1. 采用 默认 模式 捕获 SQL 语句 中 的 错误 | 
通过 prepare() 和 execute() 方 法 向 数据 库 中 添加 数据 ， 设 置 PDOStatement 对 象 的 errorCode | 
属性 ， 手 动 检测 代码 中 的 错误 。 主 要 代码 如 下 : | 


<?php 
这 $ POST['Submit] 一 "提交 " && $ POST[pdo]!=""){ | 
$dbms="mysql': // 数 据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 | 


:7367 UD 


(本 > 六 预知 道 的 300 个 问题 


库 ， 只 要 修改 它 即 可 ， 不 用 记 住 那么 多 的 函数 


$host='"localhost'; /数据 库 主机 名 
$dbName='db_database'; // 使 用 的 数据 库 
$user='root’; /数据 库 连 接 用 户 名 
S$pass="111'; // 对 应 的 密码 
$dsn="$dbms:host=$host;dbname=$dbName"; 

象 Spdo 


$query="insert into tb pdo mysqls(pdo type,database name,dates)values (".$_ POST['pdo'].", 
"$_POST['databases'].",".$_ POST['dates] 9" 
$result=$pdo->prepare($query); 
$result->execute(); 
$code=$result->errorCode(); 
这 empty($code)){ 
echo "数据 添加 成 功 !"; 
jelse{ 
echo ' 数 据 库 错 误 : <br/>'; 
echo 'SQL Query:'.$query: 
echo “'<pre>'; 
var_dump($result->errorInfo()); 


$pdo = new PDO($dsn, $user, $pass); 。“”// 初 始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连 接 对 
te 
| echo '</pre>'; 


?> 


在 本 示例 中 , 在 定义 INSERT 添加 语句 时 使 用 了 错误 的 数据 表 名 称 tb_pdo_mysqls (正确 名 
称 是 也 _pdo_mysql)， 导 臻 输出 如 图 9.3 所 示 的 结果 。 


图 9.3 在 默认 模式 中 捕获 SQL 中 的 错误 


2. 采用 警告 模式 捕获 SQL 语句 中 的 错误 
| 设置 警告 模式 通过 prepare0 和 execute() 方 法 读 取 数据 库 中 的 数据 ,并 且 通 过 while 语 名 和 fetchO 
| 方法 完成 数据 的 循环 输出 ， 体 会 在 设置 成 警告 模式 后 执行 错误 的 SQL 语句 。 其 关键 代码 如 下 : 


| $dbms='mysql'; // 数 据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 


<2php 
库 ， 只 要 修改 它 即 可 ， 不 用 记 住 那么 多 的 函数 
$host=localhost: /数据 库 主 机 名 
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$dbName='db database' /使 用 的 数据 库 
$user=Troot': /数据 库 连 接 用户 名 
Spass="111'; // 对 应 的 密码 | 
$dsn="$dbms:host-$host:dbname=$dbName": | 
ty{ | Y 
S$pdo = new PDO($dsn, $user, $pass); // 初 始 化 一 个 PDO 对 象 , 就 是 创建 了 数据 库 连 | 到 
接 对 象 Spdo | 
$pdo->setAttribute(PDO::ATTR ERRMODE,PDO:ERRMODE WARNING); /设置 为 警告 模式 
$query="select * from tb_pdo mysqls"; // 定 义 SQL 语句 | 
$result=$pdo->prepare($query); // 准 备查 询 语句 | 
$result->execute(O); // 执 行 查询 语句 ， 并 返回 结果 集 | 
while($res=Sresult->fetch(PDO::FETCH_ASSOC)){ //while 循环 输出 查询 结果 集 ， 并 且 设置 | 
结果 集 为 关联 索引 | 
?> | 
<tr> | 


<td height="22" align="center" valign="middle"><?php echo $res["id'"];?></td> 
<td align="center" valign="middle"><?php echo S$res['pdo_type'];?></td> | 
<td align="center" valign="middle"><?php echo S$res['database_name'];?></td> | 
<td align="center" valign="middle"><?php echo S$res['dates'];?></td> | 
</tr> 
| | 
} catch (PDOException $e) { 

die ("Error!: " . $e->getMessage0 . "<br/>"); 

} 

?> 


运行 结果 如 图 9.4 所 示 。 

3. 采用 异常 模式 捕获 SQL 语句 中 的 错误 

在 执行 数据 库 中 数据 的 删除 操作 时 设置 为 
异常 模式 ， 并 且 编 写 一 个 错误 的 SQL 语句 〈 操 
作 错 误 的 数据 表 tb_pdo_mysqls)， 体 会 异常 模 
式 与 警告 模式 和 默认 模式 的 区 别 。 其 关键 代码 
如 下 : 


图 9.4 设置 警告 模式 后 捕获 的 SQL 语句 错误 


<?php 
header ( "Content-type: text/html:; charset=utf-8" ); /设置 文件 编码 格式 | 
if($_GET['conn id]!-"9){ 


$dbms="mysql': /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 | 
数据 库 ， 只 要 修改 它 即 可 ， 不 用 记 住 者 么 多 | 

的 函数 | 

$host='"localhost': /数据 库 主机 名 | 
S$dbName='db database' /使 用 的 数据 库 | 
$user='root’; /| 数据 库 连接 用 户 名 | 
S$pass=111'; // 对 应 的 密码 | 
$dsn="$dbms:host=$host:dbname=$dbName"; | 
try { | 
$pdo =new PDO($dsn, $user, Spass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 | 
连接 对 象 gpdo | 

Spdo->setAttribute(PDO::ATTR. ERRMODE.PDO::ERRMODE EXCEPTION): | 
*。269 。 人 


> 人 Pp 六 须知 道 的 300 个 问题 


$query="delete from tb_ pdo_mysqls where Id=:id"; 
$result=$pdo->prepare($query); // 预 准备 语句 
Sresult->bindParam(":id',$_GET['conn id); 。”// 绑 定 更 新 的 数据 

| $result->execute(); 

| } catch (PDOException $e) { 
食 生 | | echo PDO Exception Caught.'; 
| echo 'Error with the database:<br/>"; 
ee echo 'SQL Query: "Squery; 

| echo <pre> 

| echo "Error: " . $e->getMessage(). "<br/>"; 

! echo "Code: " . $e->getCode0. "<br/>"; 
echo "File: " . $e->getFile0. "<br/>"; 
echo "Line: " . $e->getLine(). "<br/>"; 
echo "Trace: " . $e->getTraceAsString(). "<br/>"; 
echo '</pre>'; 


在 设置 为 异常 模式 后 ， 执 行 错误 的 SQL 语句 返回 的 结果 如 图 9.5 所 示 。 


?coun Ld-22 ~ icresel 


/deleb 
大 0D 


PDO Exception Canght Error with the database: 
SQL Query: delete from th_pdo_mysqls where Id 这 


图 9.5 异常 模式 捕获 的 SQL 语句 错误 信息 


| 专家 点 评 

| 由 于 异常 模式 符合 错误 处 理 的 面向 对 象 风格 (捕获 和 处 理 不 同类 型 的 异常 )， 而 且 易 于 使 
| 用 ， 因 此 大 多 数 人 会 选择 利用 PHP 更 强大 的 面向 对 象 模型 ， 并 使 用 异常 模式 处 理 错误 。 

| 无 论 选 择 哪 种 错误 处 理 方法 ， 返 回 SQL 查询 本 身 的 文本 都 是 一 个 很 好 的 想法 ， 这 将 使 用 
| 户 能 够 确切 了 解 查询 是 否 存在 问题 ， 而 且 有 助 于 进行 错误 调试 。 


问题 191 如何 向 数据 库 中 添加 或 修改 数据 ? 


问题 阐述 

| 如何 向 数据 库 中 添加 或 修改 数据 ? 
专家 解答 
可 以 使 用 SQL 中 的 INSERT 语句 向 数据 库 中 添加 数据 ， 使 用 UPDATE 语句 修改 数据 库 中 
| 已 有 的 数据 。 然 后 通过 query0 方 法 或 prepare0 和 execute0 方 法 执行 添加 和 更 新 语句 。 

应 用 示例 

| 通过 prepare() 和 execute() 方 法 向 数据 库 中 添加 数据 ， 其 具体 实现 步骤 如 下 : 

2 *。270 。 
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首先 ， 设 计 网 页 页 面 ， 创 建 form 表单 将 要 添加 的 数据 提交 到 本 页 。 然 后 ， 定 义 数 据 库 连 

接 的 参数 ， 实 例 化 PDO 构造 函数 ， 完 成 与 数据 库 的 连接 。 接 着 ， 定 义 INSERT 添加 语句 , 通 | 

过 $ POST[] 方 法 获取 表单 中 提交 的 数据 。 最 后 ， 通 过 prepare0 和 execute() 方 法 向 数据 库 中 添加 | 
数据 。 其 代码 如 下 : | 


<?php 
这 $_POST['Submit] 一 "提交 " && $_ POST[pdo]!'=""){ 
$dbms='mysql'; /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 4 
数据 库 ， 只 要 修改 它 即 可 ， 不 用 记 住 那么 多 | 
的 函数 | 
$host='"localhost'; /| 数据库 主机 名 
$dbName='db_database'; // 使 用 的 数据 库 
$user='root'; /| 数据库 连接 用 户 名 
$pass="111'; // 对 应 的 密码 
$dsn="$dbms:host=$host;:dbname=$dbName"; 
try{ 


$pdo = new PDO($dsn, $user, $pass): /初始 化 一 个 PDO 对象 , 就 是 创建 了 数据 库 连 | 
接 对 象 $pdo | 
$query="insert into tb_pdo_mysql(pdo_type,database_name,dates)values(".$ POST['pdo'].", ! 
".$_POST['databases'].",".$_POST['dates’].")"; | 
Sresult=$pdo->prepare($query); 
if($result->execute()){ 
echo "数据 添加 成 功 !"; 
}else 


{ 
echo "数据 添加 失败 !"; 


) 
} catch (PDOException $e) { 

die ("Error!: " . $e->getMessage0 . "<br/>"); 
} 


b 
> 


专家 点 评 


在 实际 的 应 用 程序 中 , 部 分 查询 输入 可 能 会 来 自 于 用 户 生成 的 内 容 。 由 于 正在 使 用 prepae0 | 
和 execute() 方 法 ， 因 此 不 必 担 心 SQL 注入 会 攻击 这 个 查询 : 所 有 变量 都 将 被 自动 转 义 。 


问题 192 ”如 何 保护 Web 站 点 免 归 SQL 注入 攻击 ? 


问题 阐述 


SQL 注入 攻击 是 指 攻 击 者 利用 站 点 中 合法 用 户 的 输入 机 制 发 送 SQL 代码 到 数据 库 执行 。 | 
避免 SQL 注入 攻击 的 黄金 法 则 是 : 在 来 自 外 部 源 的 所 有 数据 到 达 数 据 库 之 前 将 其 转 义 。 | 
则 不 仅 适用 于 INSERT 和 UPDATE 语句 ， 也 适用 于 SELECT 查询 。 

在 脚本 的 查询 中 使 用 预 编译 语句 几乎 可 以 消除 所 有 的 SQL 注入 攻击 问题 ， 但 如 果 选 择 使 | 
用 query0 方 法 ,将 不 具 必须 对 添加 到 查询 中 的 用 户 输入 手工 转 义 。 其 代码 如 下 : 


<?php 
$dbms='mysql': 


D71s L_ 


5 有 pp 如 需 知道 的 300 个 问题 


$dbName='db_database'; 


$user='Toot'; 
Spwd="111'; 
$host='localhost'; 
$dsn="$dbms:host=$host:dbname=$dbName"; 
//$name="mr'; 
$name="or Name LIKE'%": 
$query="select * from tb_zc where user=". $name .""; //SQL 语句 
UA 
$pdo=new PDO($dsn, $user,$pwd); 
$result=$pdo->query($query): /输出 结果 集中 的 数据 
foreach ( $result as $row){ /输出 结果 集中 的 数据 
ye 
<tr> 
<td bgcolor="#FFFFFF"><div align="center"><?php echo $row['user"];?>;</div></td> 
</tr> 
<?php } 
} catch (Exception $e) { 
echo "ERROR!!".$e->getMessage()."<br>"; 
} 
> 


在 这 个 示例 中 , 假定 SQL 语句 中 的 $name 变量 来 自 于 用 户 提 交 的 表单 。 通 常用 户 会 提交 像 
“mr” 这 种 内 容 ， 这 将 会 生成 下 面 的 SQL 语句 : 
select * from tb_zc Where UseI=mr 


这 在 脚本 中 不 会 造成 任何 问题 。 然 而 ， 一 个 “聪明 ”的 攻击 者 可 能 会 输入 “'OR Name 


| LIKE'%”， 从 而 得 到 下 面 的 SQL 语句 : 


select * from tb_zc where user="OR Name LIKE'%' 


这 种 输入 会 为 攻击 者 打开 整个 表 。 如 果 是 非常 敏感 的 数据 ， 将 被 完全 暴露 在 外 。 那 么 采用 


| 什么 方法 才能 解决 这 个 问题 呢 ? 
| 专家 解答 


要 想 解决 这 个 问题 ， 可 以 使 用 PDO 中 的 quote0 方 法 转 义 传递 给 SQL 字符 串 的 数据 。 只 需 


| 要 将 SQL 代码 修改 成: 


和 execute() 方 法 。 
| 专家 点 评 


| 需要” 


$sql="select * from tb_zc where user=". $pdo->quote($name) .""; 
记 住 需要 引用 SQL 查询 中 使 用 的 每 块 数据 ， 这 个 没有 快捷 途径 ， 除 非 考虑 使 用 prepare() 


如 果 正 在 使 用 PDO->query0 方 法 ,那么 总 需要 使 用 quote0 方 法 引用 用 户 的 输入 , 记 住 是 “总 


如 果 选 择 使 用 prepare0 和 execute() 方 法 ,那么 不 必 引 用 那些 需要 绑 定 到 预 编译 SQL 的 值 ( 例 


| 如 ， 将 被 插入 到 数据 库 的 值 )， 驱 动 会 为 用 户 做 所 有 的 事 。 然 而 ， 有 时 可 能 无 法 将 一 个 变量 绑 


定 到 预 编译 SQL 语句 ， 在 这 种 情况 下 ， 如 果 正 在 创建 一 个 动态 SQL 语句 ， 那 么 需要 引用 所 有 
用 到 的 无 法 被 绑 定 的 值 (如 GROUP BY 子 句 或 ORDER BY 或 表 名 称 )。 
。272 . 
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问题 193 ”如 何 获得 受 影响 的 行 数 ? 


问题 阐述 | 

在 做 进一步 动作 前 就 能 够 计算 出 一 个 查询 返回 或 影响 的 行 数 通 常 很 有 用 。 希望 分 页 显示 结 
果 或 生成 统计 信息 时 ， 这 种 功能 特别 方便 。 那 么 获取 受 影 响 行 数 的 功能 是 如 何 实现 的 呢 ? | 
专家 解答 | 

1. 通过 SQL 聚合 函数 COUNT 统计 查询 结果 记录 数 

COUNT 返回 一 个 查询 或 部 分 查询 的 行 数 ， 通 常 和 DISTINCT 一 起 使 用 。SQL 的 聚合 函数 
COUNT 被 多 种 数据 库 广泛 支持 。 

2. 通过 PDOStatement->rowCount() 方 法 返回 受 影响 行 数 

PDOStatement->rowCount() 方 法 获取 到 有 多 少 行 被 UPDATE、INSERT 或 DELETE 查询 所 | 
影响 。rowCount0 方 法 在 典型 的 PHP 应 用 程序 中 并 不 常见 ， 但 它 可 以 统计 出 在 调用 | 
PDOStatement->execute 后 ， 有 多 少 行 受 到 影响 。 | 


3， 应 用 示例 
通过 SQL 聚合 函数 COUNT 统计 查询 返回 的 记录 数 。 其 具体 代码 如 下 : 


<2php 

iftisset($_ POST['Submit]) && $ POST['Submit] 一 "查询 "){ 

$dsn = "mysql:host=localhost;:dbname=db_database;'; 

$user = Toot'; 

$password ="111'; 

$id =$_ POST[mr]: 

try{ 

$dbh = new PDO($dsn, $user, $password); | 

S$dbh->setAttribute(PDO::ATTR_ ERRMODE, | 
PDO::ERRMODE EXCEPTION); | 

$sql ='SELECT COUNT(*) FROM tb_pdo_mysql WHERE database_name=:id'; 

$stmt = $dbh->prepare($sqD); 

$stmt->bindParam('id', $id, PDO::PARAM_STR): | 

$result = $stmt->execute(); | 

echo 'There are ', $stmt->fetchColumn(), ' rows retumed.': | 


} 
catch (PDOException $e){ | 
echo 'PDO Exception Caught. '; | 
echo 'Error with the database: <br />'; | 
echo 'SQL Query: ', $sql: | 
echo 'Error: ' . $e->getMessage(); | 
} | 
> 1 


?> | 
运行 本 示例 ， 在 文本 框 中 输入 要 查询 的 数据 库 名 称 ， 单 击 “ 查 询 ” 按 钮 ， 即 可 显示 出 所 查 
询 到 的 结果 行 数 。 运 行 结果 如 图 9.6 所 示 。 | 

S373 人 


图 9.6 获取 查询 返回 的 行 数 


| 专家 点 评 


| 在 使 用 UPDATE 和 DELETE 语句 时 ， 一 定 要 随时 确定 添加 了 WHERE 子 句 。 如 果 未 添加 
| WHERE 子 句 ， 则 可 能 会 更 新 数据 库 中 的 所 有 列 或 者 删除 表 中 的 所 有 数据 ， 无 论 是 哪 一 种 情况 
| 都 不 是 用 户 希望 看 到 的 。 


问题 194 ”如 何 进行 事务 处 理 ? 


| 问题 主 述 
如 果 需 要 将 一 组 SQL 查询 作为 一 个 操作 来 运行 ， 为 了 保证 数据 的 完整 性 ， 需 要 进行 事务 处 
| 理 , 几乎 所 有 数据 库 都 以 这 样 或 那样 的 形式 提供 事务 支持 , 那么 如 何 使 用 PDO 进行 事务 处 理 呢 ? 


| 专家 解答 


在 PDO 中 实现 事务 处 理 的 功能 ， 其 应 用 的 方法 如 下 : 

回 ”开启 事务 一 一 beginTransaction() 方 法 

beginTransaction() 方 法 将 关闭 自动 提交 (autocommit) 模式 ， 直 到 事务 提交 或 回 滚 以 后 才 恢 复 。 
| 提交 事务 一 一 commit( 方 法 
| commit( 方 法 完成 事务 的 提交 操作 ， 如 果 成 功 则 返回 True， 和 否则 返回 False。 
回 ”事务 回 滚 一 rollback( 方 法 
Iollback() 方 法 执行 事务 的 回 滚 操作 。 
应 用 示例 
| 通过 prepare() 和 execute() 方 法 向 数据 库 中 添加 数据 ， 并 且 通 过 事务 处 理 机 制 确保 数据 能 够 
| 正确 地 添加 到 数据 中 。 其 具体 步骤 如 下 : 
| 创建 index.php 文件 。 首先， 定义 数据 库 连 接 的 参数 ,创建 try{}catch{} 语 句 ， 在 try 个 语句 
| 中 实例 化 PDO 构造 函数 ， 完 成 与 数据 库 的 连接 ， 并 且 通 过 beginTransaction() 方 法 开启 事务 。 
| 然后， 定义 INSERT 添加 语句 ， 通 过 $_POST[] 方 法 获取 表单 中 提交 的 数据 ， 通 过 prepare0 和 
| execute() 方 法 向 数据 库 中 添加 数据 ,并 且 通 过 commit() 方 法 完成 事务 的 提交 操作 .最 后 ,在 catch{} 
| 语句 中 返回 错误 信息 ， 并 且 通 过 rollBack() 执 行事 务 的 回 深 操 作 。 其 代码 如 下 : 


<2php 
这 $_ POST['Submit] 一 "提交 " && $_POST['pdo]!=""){ 
$dbms=mysql': /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 


| 库 ， 只 要 修改 它 即 可 ， 不 用 记 住 那 么 多 的 函数 
二 :7s 
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$host='localhost'; /数据 库 主机 名 

3$dbName= db database': /使 用 的 数据 库 

$user='root’; /数据 库 连接 用 户 名 

S$pass="111'; // 对 应 的 密码 

$dsn="$dbms:host=$host;dbname=$dbName"; 

ty{ | 
$pdo 二 new PDO($dsn, $user, $pass); 。 ”// 初 始 化 一 个 PDO 对 象 , 就 是 创建 了 数据 库 连 

接 对 象 ypdo 

$pdo->beginTransaction(): /开启 事务 


ee insert into tb_ pdo_mysql(pdo_type,database_name,dates)values(".$_ POST['pdo'].", | 
™.$_ POST['databases'].",".$_POST['dates'].")"; | 


Sresult-$pdo->prepare(Squery): 
if($result->execute()){ 
echo "数据 添加 成 功 !"; 
}else{ 
echo "数据 添加 失败 !"; 
) 
$pdo->commitO; /执行 事务 的 提交 操作 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage(0 . "<br/>"); 
$pdo->rollBackO; // 执 行事 务 的 回 深 
} 
} 
?> 
运行 结果 如 图 9.7 所 示 。 -一 全 
专家 点 评 
在 MySQL 中 ， 事 务 由 单独 单元 的 一 个 或 多 个 本 
SQL 语句 组 成 。 在 这 个 单元 中 ， 每 个 MySQL 语句 是 i 
相互 依赖 的 , 而 整个 单独 单元 作为 一 个 不 可 分 割 的 整 | | | 


体 ， 如 果 单 元 中 一 旦 某 条 SQL 语句 执行 失败 或 产生 EEC 


错误 ,整个 单元 将 会 回 滚 ,所 有 受到 影响 的 数据 将 返 ee | 
回 到 事务 开始 以 前 的 状态 ， 如 果 单元 中 的 所 有 SQL 。 图 97 数据 添加 中 应 用 事务 处 理 机 制 
语句 均 执 行 成 功 ， 则 事务 被 顺利 执行 。 但 不 是 所 有 的 数据 库 都 支持 事务 ， 因 此 ， 默 认 情 况 下 ， | 
PDO 运行 在 自动 提交 模式 下 ,在 这 种 情况 下 ,每 个 查询 被 视 为 它 自己 的 事务 进行 处 理 。 如 果 数 | 
据 库 不 支持 事务 ， 那 么 查询 执行 时 也 没有 事务 。 如 果 数 据 库 支持 事务 ， 那 么 可 以 不 使 用 自动 提 | 
交 模式 ， 而 是 使 用 手动 开始 和 停止 事务 。 | 
在 上 面 的 示例 中 , PDO-> beginTransaction0 和 Spdo->commit() 方 法 在 try 块 中 被 调用 。Catch | 
块 使 用 $pdo->rollBack() 方 法 在 出 现 问题 时 回 滚 数据 库 。 | 


问题 195 如何 利用 PDO 使 用 存储 过 程 ? 


问题 阐述 | 
许多 数据 库 支 持 存储 过 程 一 一 运行 在 数据 库 上 的 脚本 ， 通 常 使 用 数据 库 特定 的 SQL 语言 | 


:75s L_ 


本 人 Pp 六 须知 道 的 300 个 问题 


| 编写 。 存 储 过 程 允许 在 更 接近 于 数据 的 位 置 操作 数据 ， 从 而 减少 了 带宽 的 使 用 。 它 们 使 数据 独 

| 立 于 脚本 逻辑 ， 允 许 使 用 不 同 语言 的 多 个 系统 以 相同 的 方式 访问 数据 ， 从 而 节省 了 花费 在 编码 

| 和 调试 上 的 宝贵 时 间 。 另 外 ， 存 储 过 程 使 用 预定 的 执行 计划 提高 了 查询 速度 ， 并 且 能 够 阻止 与 
会 内 | 数据 的 直接 相互 作用 ， 从 而 起 到 保护 数据 的 作用 。 那 么 在 PDO 中 如 何 使 用 存储 过 程 呢 ? 


专家 解答 


1 全 建 信和 


这 里 首先 创建 一 个 存储 过 程 ， 其 SQL 语句 如 下 : 


drop procedure if exists pro_reg; 
delimiter // 

| create procedure pro_reg (in nc varchar(80), in pwd varchar(80), in email varchar(80),in address 

varchar(50)) 

| begin 

insert into tb_reg (name, pwd ,email ,address) values (nc, pwd, email, address); 

end:; 

Wh 


“drop” 语 句 删除 MySQL 服务 器 中 已 经 存在 的 存储 过 程 pro_reg。 
“delimiter //” 的 作用 是 将 语句 结束 符 更 改 为 “//”。 


2. 在 PDO 中 通过 CALL 语句 调用 创建 存储 过 程 
在 PDO 中 通过 CALL 语句 调用 存储 过 程 ， 实 现 用 户 注册 信息 的 添加 。 其 具体 步骤 如 下 : 
| 创建 index.php 文件 。 首先, 创建 form 表单 , 将 用 户 注册 信息 通过 POST() 方 法 提交 到 本 页 。 
| 然后 在 本 页 中 编写 PHP 脚本 ， 通 过 PDO 连接 MySQL 数据 库 ， 并 且 设 置 数据 库 编码 格式 为 
UTF-8， 获 取 表 单 中 提交 的 用 户 注册 信息 。 接 着 ， 通 过 call 语句 调用 存储 过 程 pro_reg， 将 用 户 
| 注册 信息 添加 到 数据 表 中 。 最 后 ， 通 过 try{}catch} 语 句 块 返回 错误 信息 。 其 关键 代码 如 下 : 


<2php 
if($_POST['submit]!=""){ 
$dbms='mysql'; /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 
库 ， 只 要 修改 它 即 可 ， 不 用 记 住 那么 多 的 函数 
$host='localhost'; /数据 库 主 机 名 
$dbName='db database': /使 用 的 数据 库 
$user=root': /数据 库 连接 用 户 名 
$pass='111': /对 应 的 密码 
$dsn="$dbms:host=$host'dbname=$dbName'"': 
fry 
$pdo = new PDO($dsn, $user, $pass); // 初 始 化 一 个 PDO 对 象 , 就 是 创建 了 数据 库 连 
接 对 象 $pdo 
S$pdo->query("set names utfg"): // 设 置 数据 库 编码 格式 
$pdo->setAttribute(PDO::ATTR ERRMODE.PDO::ERRMODE EXCEPTION): 
// 定 义 错误 异常 模式 


$nc=$_POST['ne']; 
S$pwd=md5($_POST['pwd"]); 
$email-$ POST['email]: 
$address=$ POST['address']: 


了 。276 。 
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$query="call pro_reg('$nc','$pwd','$email','$address")":; | 
Sresult=$pdo->prepare($query); | 
if($result->execute()){ | 

echo "数据 添加 成 功 !"; | 
jelse{ | 


echo "数据 添加 失败 !"; 


} catch (PDOException $e) { 


echo 'PDO Exception Caught."; | 
echo 'Error with the database:<br/>' ! 
echo 'SQL Query: '.$query; | 
echo '<pre>'; 

echo "Error: " . $e->getMessage(). "<br/>"; | 
echo "Code: " . $e->getCode0. "<br/>"; | 
echo "File: " . $e->getFile0. "<br/>"; | 
echo "Line: " . $e->getLine0. "<br/>"; 

echo "Trace: " . $e->getTraceAsString(). "<br/>"; 


在 MySQL 中 ,创建 存储 过 程 的 基本 形式 如 下 : 


CREATE PROCEDURE sp_name ([proc_parameter[,...]]) 
[characteristic...] routine body 


echo '</pre>'; | 

) 

> | 
运行 结果 如 图 9.8 所 示 。 | 
Rr | 

用 户 昵 称 ; |Imrson | 

注册 密 码 } [| | 

Et [CR | 

穿 应 性 让: FE 和 | 

注册 | 重 三 | 

图 9.8 通过 存储 过 程 完成 用 户 的 注册 | 

专家 点 评 | 


参数 说 明 : 

@ sp_name: 表示 存储 过 程 的 名 称 ; 

@ proc_parameter: 表示 存储 过 程 的 参数 列表 ; 
@ characteristic: 表示 指定 存储 过 程 的 特性 ; 
@ routine body: 表示 SQL 代码 的 内 容 。 | 


说 明 : 

proc_parameter 中 的 参数 由 3 部 分 组 成 ， 分 别 是 输入 输出 类 型 、 参 数 名 称 和 参数 类 型 。 其 | 
形式 为 [ IN | OUT |INOUT ]param name type。 其 中 ，IN 表示 输入 参数 ; OUT 表示 输出 参数 ; 
INOUT 表示 既 可 以 输入 也 可 以 输出 ; param name 参数 是 存储 过 程 的 参数 名 称 ; type 参数 指定 
存储 过 程 的 参数 类 型 ， 该 类 型 可 以 为 MySQL 数据 库 的 任意 数据 类 型 。 
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MySQL 中 存储 过 程 的 建立 以 关键 字 create procedure 开始 ， 后 面 仅 跟 存储 过 程 的 名 称 和 参 
| 数 。MySQL 的 存储 过 程 名 称 不 区 分 大 小 写 ， 如 PROCE10 和 proce10 代 表 同 一 存储 过 程 名 。 存 
| 储 过 程 的 名 称 不 能 与 MySQL 数据 库 中 的 内 建 函 数 重 名 。 
信和 | MySQL 存储 过 程 的 语句 块 以 begin 开始 ， 以 end 结束 。 语句 体 中 可 以 包含 变量 的 声明 、 控 
= 一 | 制 语句 、SQL 查询 语句 等 。 由 于 存储 过 程 内 部 语句 要 以 分 号 结束 ， 所 以 在 定义 存储 过 程 前 ， 应 
将 语句 结束 标志 “;” 更 改 为 其 他 字符 ， 并 且 应 降低 该 字符 在 存储 过 程 中 出 现 的 机 率 ， 更 改 结 束 
示 志 可 以 用 关键 字 “delimiter” 定 义 ， 例 如 : 
mysql>delimiter // 
存储 过 程 创 建 之 后 ， 可 用 如 下 语句 进行 删除 ， 参 数 proc_name 指 存储 过 程 名 。 
drop procedure proc_name 


问题 196 ”如 何 利用 PDO 连接 Oracle llg? 


| 问题 阐述 

| 在 PHP 中 ， 如 何 利用 PDO 来 连接 Oracle 11g 数据 库 呢 ? 

| 专家 解答 

| 通过 PDO 连接 Oracle 11g 数据 库 ， 首 先 要 在 本 机 上 安装 Oracle 118 的 客户 端 。 然 后 ， 在 
| php.ini 文件 中 加 载 extension=php_pdo.dll 和 extension-php_pdo_oci.dll 模块 .最 后 ,将 Oracle 11g 
| 客户 端的 oraocieil1.dll、oci.dll 和 orannzsbb11.dll 复制 到 Apache 的 bin 文件 夹 下 。 

| 本 
| 切记 在 完成 上 述 配 置 操作 后 ,应 重新 启动 Apache 服务 器 ,之 后 就 可 以 通过 PDO 连接 Oracle 
| 11g 数据库 了 。 

| 应 用 示例 

| 下 面 连接 Oracle 数据 库 服务 器 ,数据 库 名 称 是 192.168.1.188:1521/orcl, 数据 库 服务 器 的 用 
| 户 名 是 system， 密 码 是 mrsoft。 

| 具体 操作 步 又 为 ， 首 先 定义 数据 库 连 接 的 参数 ， 然 后 实例 化 PDO 构造 函数 ， 最 后 通过 try 
| catch 语句 判断 数据 库 是 否 连 接 成 功 。 具 体 代码 如 下 : 


<?php 

header("Content-type:text/html; charset=utf-8"): 

$dbms="oci'; /| 数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 
库 ， 只 要 修改 它 即 可 ， 不 用 记 住 那么 多 的 函数 

$dbName="192.168.1.188:1521/orcl'; // 使 用 的 数据 库 

$user='system'; /数据 库 连 接 用 户 名 

$pass="mrsoft'; /对 应 的 密码 

$dsn="$dbms:dbname=$dbName'": 

ty { 

$pdo = new PDO($dsn, $user, $pass); // 初 始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连接 对 

象 Spdo 


echo "Oracle 11g 连接 成 功 !<br/>"; 
了 gf 
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$pdo = null; 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 


} 
> 
连接 测试 结果 如 图 99 所 示 。 rr 
ee OO Er Blox 有 # 
专家 点 评 || 安 业 6 夫 | 看 hnpy//12700.probl. 价 " 目 > 


Oncke 11g 连 全 或 几 


通过 PDO 连接 Oracle 数据 库 ， 连 接 Oracle 10g 和 | 
连接 Oracle 11g 的 方法 是 相同 的 ， 都 必须 在 本 机 上 安装 - | 
Oracle 数据 库 的 客户 端 ， 然 后 将 客户 端 中 的 oraocieilldll、 图 99 连接 Oracle lls 数据 库 | 
ocidll 和 orannzsbb11.dll 复制 到 Apache 的 bin 文件 夹 下 。 | 


问题 197 如 何 封装 PDO 数据 库 连 接 类 ? 


EB @ imernet | PR 有 本 = ai 。 | 
- 


问题 阐述 | 


虽然 PDO 是 数据 库 抽象 层 ， 可 以 连接 、 操 作 不 同 的 数据 库 ， 但 是 就 其 连接 方法 而 言 ， 不 
同 的 数据 库 之 间 仍 然 存在 一 些 区 别 ， 那 么 如 何 才 能 忽略 这 些 区 别 ， 达 到 真正 的 统一 呢 ? 


专家 解答 


要 想 达到 这 种 真正 的 统一 ， 最 有 效 的 方法 就 是 将 数据 库 的 连接 方法 封装 到 类 中 ,直接 向 类 | 

中 传递 不 同 的 参数 ， 即 可 实现 与 不 同 数据 库 的 连接 。PDO 数据 库 连接 类 ConnDB 的 语法 如 下 : | 

/六 闵 六 六 浆 闵 闵 浆 六 率 交 六 这 六 六 率 闵 六 六 六 冰冰 闲 六 六 六 六 闲 六 六 闵 闵 永 六 六 六 六 率 闵 闲 冰 冰冰 冰冰 闲 | 

* @ 说 明 ; PDO 连接 数据 库 类 | 

* @ 作者 : 明日 科技 | 

*@ 学 习 社 区 : www.mrbccd.com | 

* @ E-mail:mingrisoft@mingrisoft.com | 

六 来 闵 率 素 六 水 浆 闵 六 六 六 水 闵 素 六 六 六 六 六 六 六 六 率 闵 六 六 冰 闵 六 六 六 闵 六 六 六 六 六 率 率 冰 闵 亲 六 永 冰 / | 

class ConnDB{ | 

Var $dbtype; 
Var $host:; 


* @ 方法 说 明 : 
* ”构造 方法 ， 类 的 初始 化 | 


来 

* @ 参数 说 明 : | 
* ”Sdbtype: 连接 数据 库 的 类 型 | 
* $host; 数据 库 服务 器 主机 名 或 卫 地 址 | 
* ”$user: 用 户 名 

* ”S$pwd: 密码 

* ”$dbname: 数据 库 名 称 
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function ConnDB($dbtype,$host, $user,$pwd,$dbname){ 
| Sthis->dbtype=$dbtype: 
| $this->host=$host: 
| Sthis->user=$user; 
S$this->pwd=$pwd: 
S$this->dbname=$dbname; 
. 
*@ 方法 说 明 : 
* 获取 数据 库 连 接 ID 
wi 
function GetConnIdO{ 
if($this->dbtype—"mysql" | $this->dbtype—"mssql"){ // 判 断 数据 库 的 类 型 ， 根 据 类 
型 定义 DSN 的 值 
$dsn="$this->dbtype:host=$this->host:dbname=$this->dbname"; 
jelse{ 
$dsn="$this->dbtype:dbname=$this->dbname"; 


! 
ty { 
$conn = new PDO($dsn, $this->user, $this->pwd); /初始 化 PDO 对 象 , 创建 数据 
库 连接 对 象 $pdo 
$conn->query("set names utf8"); /设置 编码 格式 
return $conn; // 返 回 数据 库 连 接 标识 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage0 . "<br/>"); 
} 
} 
] 


通过 PDO 连接 MySQL 数据 库 ， 数 据 库 名 称 为 db_database， 用 户 名 为 root， 密 码 为 111。 
具体 操作 步骤 为 : 首先 封装 PDO 连接 数据 库 类 system.class.inc php， 然 后 创建 system.inc.php 
文件 ， 载 入 system.class.inc.php 文件 ， 实 例 化 数据 库 连接 类 ConnDB， 返 回 连接 标识 $conn。 
system.inc.php 文件 的 代码 如 下 : 


<2?php 

require("system.class.inc.php"); // 包 含 数据 库 连 接 和 操作 类 
$connobj=new ConnDB("mysql","localhost","root","111"," db_database"): /数据 库 连 接 类 实例 化 
$conn=$connobj->GetConnId0: /执行 连接 操作 ， 返 回 连接 标识 
es 


问题 198 ”如 何 封装 PDO 数据 库 操作 类 ? 


| 问题 阐述 

| 既然 可 以 对 PDO 连接 数据 库 的 方法 进行 封装 , 那么 也 就 可 以 对 PDO 操作 数据 库 的 方法 进 
| 行 封装 ， 而 这 个 PDO 操作 数据 库 的 方法 是 如 何 实现 的 呢 ? 

| 专家 解答 

PDO 数据 库 抽象 层 对 数据 库 的 操作 方法 进行 了 封装 , 实现 对 不 同 数据 库 的 统一 操作 ,管理 。 
J "280 . 
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在 我 们 封装 的 PDO 操作 数据 库 类 中 ， 它 不 但 是 不 同 数据 库 操作 的 集成 ， 而 且 还 可 以 对 参数 传 
递 的 SQL 语句 进行 判断 ， 进 而 根据 不 同 的 语句 执行 不 同 的 操作 ， 并 且 返 回 结 果 。PDO 数据 库 ， 
操作 类 AdminDB 的 语句 如 下 : | 
[让 本 本 汪汪 训 吾 沁 攻 家 下 吉 学 下 兴 光 玫 环 划 从 可 汪 业 束 区 本 全 池 业 衣 光 汪 兴 家 永光 机 这 攻 昌 吉 汪 束 
* @ 说 明 : PDO 操作 数据 库 类 
* @ 作者 : 明日 科技 
*@ 学 习 社 区 : wwwmrbccd.com 
* @ E-mail:mingrisoft@Wmingrisoft.com | 
六 六 六 水 六 六 六 来 水 六 冰冰 六 水 六 六 水 六 六 闵 六 这 六 六 六 六 六 六 浆 率 六 冰冰 六 水 率 闵 冰冰 永 率 六 冰冰 冰冰 这/ 
class AdminDB{ ! 
全 | 
* @ 方法 说 明 ; | 
* 执行 SQL 语句 | 
本 
* @ 参数 说 明 : 
* ”$sql: 所 要 执行 的 SQL 语句 
* ”Sconn: 数据 库 连 接 ID 
a 
function ExecSQL($sqlstr, $conn){ 
$sqltype=strtolower(substr(trim($sqlstr),0,6)); 
S$rs=$conn->prepare($sqlstr); // 准 备查 询 语句 
S$rs->execute(); /执行 查询 语句 ， 并 返回 结果 集 
if($sqltype—"select"){ 
$array=$Is->fetchAll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 数据 
这 count($array) 一 0 || $rs=—false) 


Tetum false; /返回 结果 
else 
return $array: /返回 数组 结果 集 
}elseif ($sqltype 一 "update" || $sqltype 一 "insert" || $sqltype 一 "delete"){1V/ 如 果 是 此 类 语句 
if($rs) 
retum true; /执行 成 功 返 回 true 
else 
Tetum false; /否则 返回 false 


1 
} 
} 


应 用 PDO 操作 数据 库 类 ， 查 询 db_database 数据 库 中 tb_commo 数据 表 中 的 数据 ， 并 且 输 | 

出 查询 结果 。 具 体操 作 步 又 为 ， 首 先 实例 化 PDO 数据 库 操作 类 AdminDB， 然 后 通过 实例 化 对 | 

象 Sadmindb 调用 ExecSQL0 方 法 ， 查 询 tb_bigtype 数据 表 中 的 数据 ， 最 后 输出 查询 结果 。 其 关 | 

键 代码 如 下 : | 

$admindb=new AdminDB0: /数据 库 操作 类 实例 化 

S$res = $admindb->ExecSQL("select * from tb commo ",$conn.1,1"); /执行 查询 语句 | 

for($i=0:$i<count($res):$i++){ | 

<img src="<?php echo $res[$il[pics']:?>” width="90" height="100" alt="<?php echo S$res[$i] | 

[name'];?>" style="border: 1px solid #f0f0f0:" /> | 
1 


281s 


人 Pp 六 须知 道 的 300 个 问题 


问题 199 ”如何 封装 PDO 分 页 类 ? 


鲜 | 。 问 题 间 述 


| 数据 库 查询 出 的 结果 集 进行 分 页 输出 呢 ?” 是 否 可 以 将 分 页 输 
专家 解答 


出 分 页 的 超 链接 。SepPage 类 的 语法 如 下 : 
/让 本 冰 束 事 束 束 事 束 可 求 束 事 事 事 可 可 于 事 事 束 来 事 事 可 可 可 事 可 事 束 事 束 事 事 训 本 束 束 可 束 束 可 可 可 事 束 可 
*@ 说 明 : PDO 分 页 类 
* @ 作者 : 明日 科技 
*@ 学 习 社 区 : www.mrbccd.com 
* @ E-mail:mingrisoft@mingrisoft.com 
六 六 闵 水 闵 六 六 来 水 闵 六 闵 米 洲 六 六 米 水 素来 水 六 六 六 六 六 六 六 率 闵 闵 闵 六 水 素 闵 水 六 水 率 玉 闵 六 六 来 冰冰/ 
// 分 页 类 
class SepPage{ 
Var $rs; 
Var $pagesize; 
Var $nowpage; 
Var $array; 
Var $conn; 
Var $sqlstr; 
es 
*@ 方法 说 明 : 
* ”数据 分 页 处 理 ， 并 以 数组 形式 返回 分 页 信息 


来 

* @ 参数 说 明 : 

* $sqlstr: 所 要 执行 的 SQL 语句 

* ”$conn: 数据 库 连接 ID 

* ”S$pagesize: 每 页 显示 数据 的 条 数 

* ”Snowpage: 当前 显示 的 页 数 

| Gad 

function ShowData($sqlstr,$conn,$pagesize, $nowpage){ 
if(!isset($nowpage) || $nowpage—"") 

S$this->nowpage=1; 

else 

| S$this->nowpage=$nowpage; 

S$this->pagesize=$pagesize; 

| S$this->conn=$conn; 

| S$this->sqlstr=$sqlstr; 

| S$offset=($this->nowpage-1)*$this->pagesize; 

| $sql=S$this->sqlstr." limit $offset, $this->pagesize": 

Sresult=$this->conn->prepare($sq]); 

$result->execute(); 
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连接 、 操 作 数 据 库 的 最 终 目 的 是 数据 的 添加 、 更 新 、 删 除 及 查询 。 那 么 如 何 将 PDO 操作 


封装 SepPage 类 ， 调 用 PDO 操作 数据 库 的 方法 ， 实 现 对 数据 库 中 数据 的 分 页 查询 ， 并 输 


// 定 义 方法 
/判断 变量 值 是 否 为 空 
/定义 每 页 起 始 页 


/定义 每 页 输出 的 记录 数 
/连接 数据 库 返 回 的 标识 
// 执 行 的 查询 语句 


// 准 备查 询 语句 
// 执 行 查询 语句 ， 并 返回 结果 集 


S$this->array=$result->fetchAll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 数据 
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这 count($this->array)- 一 0 || $this->array—false) 
Tetum false; 
else 
Teturm $this->array: 
} 
/* 
*@ 方法 说 明 : 
* ”创建 数据 分 页 输出 的 超 链 接 


*@ 参数 说 明 : 


$str.=$contentname."&nbsp;".$record."&nbsp:".$utits."&nbsp; 每 页 &nbsp:".$this->pagesize." 
&nbsp;".$utits."&nbsp; 第 &nbsp:".$this->nowpage."&nbsp; 页 / 共 &nbsp;".$pagecount."&nbsp:; 页 "; 
$str.="&nbsp;&nbsp;&nbsp:;&nbsp;"; 
if($this->nowpage!=1) 
$str.="<a 
href=".$_SERVERI['PHP_SELF']."?page=l&page type=".$anothersearchstr."&parameter2=".$anothersearchstrs. 
"class=".$class."> 首 页 </a>"; 
else 
$str.="<font color='#555555 人 首页 </font>"; 
$str.="&nbsp:"; 
if($this->nowpage!=1) 
$str.="<a 
href=".$_SERVER['PHP_SELF']."?page=".($this->nowpage-1)."&page type=".$anothersearchstr."&parameter2=". 
$anothersearchstrs." class=".$class."> 上 一 页 </a>"; 
else 
$str.="<font color=#555555> 上 一 页 </font>"; 
$str.="&nbsp:"; 
if($this->nowpage!=$pagecount) 
$str.="<a 
href=".$_SERVERI['PHP SELF']."?page=".($this->nowpage+1)."&page type=".$anothersearchstr."&parameter2=". 
$anothersearchstrs." class=".$class."> 下 一 页 </a>"; 
else 
S$str.="<font color=#555555 作 下 一 页 </font>"; 
$str.="&nbsp;"; 
if($this->nowpage!=$pagecount) 
$str.="<a 
href=".$_SERVERI['PHP_ SELF']."?page=".$pagecount."&page type=".$anothersearchstr."&parameter2=".$anot 
hersearchstrs." class=".$class."> 尾 页 </a>"; 
else 


* ”Scontentname: 超 链接 总 称 | 
* S$utits: 超 链接 中 数据 的 单位 | 
* ”Sanothersearchstr: 超 链接 传递 的 第 一 个 参数 | 
* $anothersearchstrs: 超 链接 传递 的 第 二 个 参数 | 
* Sclass 超 链 接 传递 的 样式 | 
*/ | 
function ShowPage($contentname, $utits,$anothersearchstr,$anothersearchstrs,$class){ | 
St | 
Sres=$this->conn->prepare($this->sqlstr); // 准 备查 询 语句 | 
Sres->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 | 
$this->array=$res->fetchAll(PDO::FETCH ASSOC); /获取 结果 集中 的 所 有 数据 ! 
S$record=count($this->array); // 统 计 记 录 总 数 | 
S$pagecount=ceil($record/$this->pagesize); /计算 共有 几 页 | 
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Ee 
CS 
$str.="<font color=#555555 作 尾 页 </font>"; 
这 count($this->array) 一 0 || $this->array—false) 
Tetum "无 数据 ! "; 
else 
Tetum $str; 
} 
上 


应 用 PDO 分 页 类 ， 对 db_database 数据 库 中 tb_commo 数据 表 中 的 数据 进行 分 页 输出 。 载 
入 类 的 实例 化 文件 , 通过 分 页 类 AdminDB 的 Sadmindb 对 象 调用 ShowData () 方 法 ,完成 数据 表 
中 数据 的 分 页 查询 操作 ， 调 用 ShowPage() 方 法 输出 分 页 超 链 接 。 其 关键 代码 如 下 : 
$connobj=new ConnDB("mysql","localhost","root","111","db_database"); /数据 库 连 接 类 实例 化 


$conn=$connobj->GetConnId0; /执行 连接 操作 ， 返 回 连接 标识 
$admindb=new AdminDB(): /数据 库 操作 类 实例 化 
$seppage=new SepPage(); /分 页 类 实例 化 
ifisset($_GET[page])){ /判断 分 页 变量 是 否 存在 
$page=$_GET[page']: 
Jelse{ 
S$page=1; 
1 
S$res = $seppage->ShowData("select * from tb_commo ",$conn,2,$page); /执行 查询 语句 
for($i=0:$i<count($res):$i++){ /循环 输出 查询 结果 
<table width="636" height="134" border="0" cellspacing="0" cellpadding="0"> 
<tr> 


<td width="145" rowspan="5" align="center" valign="middle"><img src="<?php echo 
S$res[$i]['pics']:?>" width="90" height="100" alt="<?php echo S$res[$il['name'];?>" style="border: lpx solid 
#f0f0f0;" /></td> 

<td height="35"> 商 品名 称 : <?php echo $res[$i]['name']:?></td> 

<td width="156" height="35"> 商 品类 别 : <?php echo $res[S$i]['class"]:?></td> 

<td width="157"> 商 品 型 号 : <?php echo Sres[$il[model]:?></td> 

</tr> 
<tr> 

<td height="25"><?php echo $seppage->ShowPage(" 产 品 "," 个 ",$_GET ['page_type'], 

"a");9></td> 
</tr> 
</table> 
<2php 


问题 200 ”如何 封 装 ADODB 连接 数据 库 类 ? 


问题 阐述 

连接 数据 库 类 的 封装 意味 着 通过 ADODB 连接 不 同 的 数据 库 的 方法 更 加 简单 , 更 容易 更 新 
和 修改 。 只 要 向 类 中 传递 对 应 的 参数 值 ， 即 可 完成 与 不 同 数据 库 服务 器 的 连接 。 那 么 如 何 实现 
ADODB 连接 数据 库 类 的 封装 呢 ? 
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ADODB 连接 数据 库 系 统 使 用 ADONewConnection0 函 数 ， 连 接 数 据 库 如 果 选 择 持久 化 连 

接 ， 则 使 用 PConnect(O) 函 数 ， 如 果 选 择 非 持久 化 连接 ， 则 使 用 ConnectO 函 数 。 | 

(1) 使 用 ADONewConnection() 函 数 连接 数据 库 系 统 。 其 语法 如 下 : 
ADONewConnection($databaseType) 


参数 $databaseType: 表示 要 连接 的 数据 库 系统 的 名 称 ， 例 如 “mysql”、“mssql” 等 。 
(2) 使 用 PConnect0) 函 数 实现 与 数据 库 持久 化 连接 。 其 语法 如 下 : 
PConnect($host,[$user],[$password],[$database]) 
参数 说 明 : 
@ $host 数据 库 系统 的 服务 器 所 在 地 址 。 如 果 是 本 机 操作 ， 则 参数 值 为 localhost。 
@ $user: 数据 库 用 户 名 。 
日 $password: 数据 库 密码 。 
@ $database: 使 用 到 的 数据 库 。 
(3) 使 用 Connect0 函 数 实现 与 数据 库 的 非 持久 化 连接 。 其 语法 如 下 : 
Connect($host,[$user],[$password],[$database]) 


参数 说 明 : 

@ $host: 数据 库 系统 的 服务 器 所 在 地 址 。 如 果 是 本 机 操作 ， 则 参数 值 为 localhost。 

@ $user: 数据 库 用 户 名 。 

日 $password: 数据 库 密码 。 

@ $database: 使 用 到 的 数据 库 。 | 

掌握 连接 数据 库 服务 器 所 使 用 的 函数 之 后 ， 就 是 定义 类 、 方 法 和 参数 ， 完 成 与 指定 数据 库 | 
服务 器 的 连接 ， 并 返回 连接 标识 。 | 

应 用 示例 

通过 AODOB 封装 数据 库 连接 类 ， 实 现 与 MySQL 数据 库 的 连接 。 其 具体 实现 步骤 如 下 : “| 

(1) 首先 创建 found.database php 文件 ,定义 ConnDB 连接 数据 库 类 , 定义 ConnDB( 方 法 | 
为 成 员 变量 赋值 ， 然 后 定义 GetConnId() 方 法 实现 与 不 同 数据 库 服务 器 的 连接 ， 并 返回 连接 标 | 
识 ， 最 后 定义 CloseConnId0 方 法 关闭 数据 库 连 接 。 其 关键 代码 如 下 : | 


<2php 
// 数 据 库 连接 类 
class ConnDB{ 
var $dbtype; /| 数据 库 服务 器 类 型 
Var $host; // 服 务 器 名 称 
Var $user; /用户 名 
Var $pwd: /密码 
Var $dbname; /数据 库 名 称 | 
var $debug; // 是 否 返 回执 行 的 SQL 语句 ， 默 认 设 置 为 返回 SQL | 
Var $conn; // 连 接 数据 库 时 返回 的 连接 标识 | 


function ConnDB($dbtype,$host, $user, $pwd,$dbname,$debug=true){ 

// 构 造 方法 ， 为 成 员 变 量 赋值 

S$this->dbtype=$dbtype:; 
S$this->host=$host; 
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S$this->user=$user; 

| Sthis->pwd=$pwd; 

| S$this->dbname=$dbname; 
| $this->debug=$debug: 


) 
function GetConnIdO{ /实现 与 不 同 数据 库 的 连接 并 返回 连接 对 象 
require("../adodb5/adodb.inc.php"); // 调 用 ADODB 类 库 文件 
if($this->dbtype—"mysql" || $this->dbtype—"mssql"){ 
// 判 断 成 员 变 量 传递 的 数据 库 类 型 
if($this->dbtype—"mysql") // 判 断 如 果 是 MySQL 数据 库 
S$this->conn=NewADOConnection("mysql"); 
// 执 行 与 MySQL 数据 库 的 连接 
else 
S$this->conn=NewADOConnection("mssql"); 
S$this->conn->Connect($this->host, $this->user,$this->pwd,$this-> 
dbname); // 数 据 库 连接 的 用 户 、 密 码 
}elseif($this->dbtype—"access"){ // 判 断 如 果 使 用 的 是 Access 数据 库 
S$this->conn=NewADOConnection("access"); 
S$this->conn->Connect("Driver={Microsoft Access Driver (*.mdb)}: 
Dbq=".$this->dbname.";Uid=".$this->user.";Pwd=".$this->pwd.";"); /执行 连接 Access 数据 库 
} 
$this->conn->Execute("set names utf8"); 。“”// 设 置 数 据 库 的 编码 格式 
这 $this->dbtype 一 "mysql) 
S$this->conn->debug=$this->debug; 
return $this->conn; // 返 回 连 接 对 象 


} 
function CloseConnIdO{ // 定 义 关 闭 数据 库 的 方法 
S$this->conn->Disconnect(): // 执 行 关 闭 的 操作 
b 
bp 


?> 
(2) 创建 connect.database.php 文件 ， 实 例 化 数据 库 连 接 类 ConnDB， 并 向 类 中 传递 对 应 
的 参数 ， 完 成 与 MySQL 数据 库 服务 器 中 db_database 数据 库 的 连接 。 其 关键 代码 如 下 : 


<?php 

require("found.database.php"): // 包 含 类 文件 

$connobj=new ConnDB("mysql","localhost","root"."111","db_database",false); /数据 库 连 接 类 实例 化 
$conn=$connobj->GetConnId(); // 返 回 连接 标识 

?> 


| (3) 创建 index php 文件 ， 包 含 数据 库 连接 类 的 实例 化 文件 ， 根 据 返回 的 连接 标识 判断 数 
| 据 库 是 否 连接 成 功 。 其 关键 代码 如 下 : 


| <2php 

| include_once 'conn/connect database php' // 包 含 数据 库 连接 类 的 实例 化 文件 
| if($conn){ 

| echo "<script>alert(' 数 据 库 连接 成 功 ! ):windowjlocation href- http://wwwmrbccd.com':</script>": 
| jelse{ 

| echo "<script>alert( 数据库 连接 失败 ! ");window.location.href='index.php';</script>"; 

| bi 

| ?> 
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运行 本 示例 ,连接 MySQL 数据 库 服 务 器 ,选择 db_database 数据 库 ， 其 运行 结果 如 图 9.10 
所 示 。 | 


图 9.10 ADODB 连接 数据 库 类 


专家 点 评 


定义 连接 数据 库 类 ,在 定义 连接 方法 GetConnIdO 时 ,必须 先 通过 require0 语 句 包含 ADODB | 
类 库 文件 adodb.incphp， 和 否则 其 他 一 切 操作 都 无 从 谈 起 。 | 
建议 在 程序 的 开发 阶段 ,在 运用 连接 数据 库 操作 类 时 ， 将 参数 Sdebug 的 值 设置 为 True, 这 | 
样 可 以 返回 执行 的 SQL 语句 ， 便 于 获取 SQL 语句 在 执行 过 程 中 出 现 的 错误 。 如 果 是 在 生产 环 | 
境 中 运用 连接 数据 库 类 ， 那 么 就 要 将 $debug 的 值 设置 为 False。 | 


问题 201 ADODB 操作 结果 集 的 方法 有 哪些 ? 


问题 阐述 | 
通过 ADODB 类 库 操作 数据 库 , 不 但 要 掌握 数据 库 的 连接 方法 ， 并且 还 要 掌握 操作 结果 集 | 
的 方法 。 那 么 ADODB 操作 结果 集 有 哪些 方法 呢 ? | 
专家 解答 | 
当 使 用 execute0 函 数 执 行 SQL 指令 时 ， 会 回 传 一 个 ADORecordSet 对 象 。 通 过 对 | 
ADORecordSet 对 象 的 控制 ， 可 以 对 结果 集 进行 很 多 操作 。 其 操作 结果 集 的 方法 如 下 : | 
1. fields 
该 变量 保存 的 是 当前 指针 所 指向 的 记录 。 
2. EOF | 
该 变量 记录 当前 指针 是 否 指向 最 后 一 条 记录 。 如 果 是 最 后 一 条 记录 ， 则 设 定 为 True; 否则 | 
为 False。 | 
3. GetArray([$number of rows]) | 
返回 从 当前 指针 指向 的 记录 开始 ， 到 ($number_of rows - 1) 行 的 全 部 记录 的 数组 。 参 数 | 
S$number of rows 是 指定 的 记录 行 。 如 果 没 有 给 出 ， 则 一 直到 EOF 才 停止 。 | 
4. MoveNext() | 
函数 作用 : 将 ADORecordSet〈 结 果 集 ) 的 指针 下 移 一 位 。 如 果 成 功 ， 则 返回 True， 和 否则 | 
返回 False。 | 
5. Move($to) 


函数 作用 : 将 ADORecordSet (结果 集 ) 的 指针 移动 到 指定 位 置 。 如 果 $to 等 于 0， 则 指针 
指向 结果 集 的 第 一 条 数据 ; 如 果 $to 的 值 大 于 结果 集 ， 则 指针 指向 最 后 一 条 数据 。 注 意 : 这 里 | 
.287 . ! 


| MoveNext(O) 混 用 。 
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6. MoveFirst() 
函数 作用 : 指针 移动 到 第 一 条 数据 。 等 同 于 Move(0)。 


7. MoveLast() 

函数 作用 : 指针 移动 到 最 后 一 条 数据 。 等 同 于 Move(RecordCountO - 1)。 

8. FetchRow() 

函数 作用 : 返回 当前 指针 指向 的 记录 的 数组 ， 如 果 是 EOF， 回 传 False。FetchRow 不 要 和 


9. FetchField($column number) 

函数 作用 : 返回 一 个 对 象 ， 包 含 着 字段 $column_number 的 名 称 、 空 间 长 度 等 相关 信息 。 
参数 $column_number 是 要 查看 的 字段 名 称 。 

10. FetchNextObject([$toupper=true]) 

函数 作用 : 返回 当前 指针 所 指向 的 记录 的 对 象形 式 ， 并 且 指 针 自动 下 移 一 行 。 

参数 $toupper 如 果 等 于 True， 则 字段 名 为 大 写 形式 。 

11. FieldCountO 

函数 作用 : 返回 结果 集中 的 字段 数 。 

12. RecordCount( ) 

函数 作用 : 返回 结果 集中 的 记录 数 。 

13，MetaType($nativeDBType [.$field max length],[Sfieldobj]) 

函数 作用 : 标准 化 不 同 数据 库 系统 下 的 数据 类 型 的 表示 法 。 处 理 不 同 的 数据 库 系 统 有 一 个 


| 问题 : 每 一 个 数据 库 系统 对 于 相同 的 数据 类 型 会 有 不 同 的 表示 法 ， 如 支持 超 长 文本 的 类 型 ， 有 


的 表示 为 text， 有 的 表示 为 long character， 还 有 的 表示 为 clob 等 。MetaType() 函 数 可 以 将 这 几 
种 表示 法 统一 起 来 。 如 用 大 写字 母 D 来 表示 所 有 的 日 期 类 型 ， 用 大 写字 母 T 表示 所 有 的 时 间 
类 型 。 
参数 说 明 : 
@ $nativeDBType: 数据 表 中 的 数据 类 型 。 
@ $fields_max_length: 字段 最 大 长 度 。 
@ Sfieldsobj: 字段 对 象 。 
MetaType(0 支 持 的 标准 化 数据 类 型 有 : 
: 支持 char、varchar 等 字符 类 型 
: 支持 text、long character 等 长 类 型 。 
: blob 或 binary image。 
: 日 期 date。 
: 时 间 timestamp。 
: 逻辑 类 型 和 位 类 型 。 
: 包含 编号 、 整 型 、 浮 点 型 等 数字 类 型 。 
: 包含 序列 、 自 动 增加 整数 等 序列 类 型 。 
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应 用 示例 | 
在 本 示例 中 应 用 ADODB 类 库 中 操作 结果 集 的 方法 ， 循 环 输出 数据 表 中 的 字段 和 值 。 其 具 
体 步骤 如 下 : 
(1) 创建 conn 文件 夹 ， 编 写 conn.php 文件 ， 载 入 adodb.iniphp 文件 。 实 现 与 MySQL 政 | 优 六 
据 库 服务 器 中 db_database 数据 库 的 连接 ， 并 设置 数据 库 的 编码 为 UTF-8。 | 
(2) 创建 ndex php 文件 ， 包 含 数据 库 连接 文件 ， 定 义 SQL 查询 语句， 通过 execute0 函 
数 执行 查询 操作 , 通过 FieldCount0 函 数 统计 字段 数量 , 应 用 for 语句 和 FetchField0 函 数 循环 输 | | 
出 各 个 字段 的 值 ， 并 应 用 metaTypeO 函 数 对 字段 中 的 数据 类 型 进行 判断 ， 同 时 应 用 DBpae0 | 
函数 对 时 间 进 行 格式 化 。 其 关键 代码 如 下 : | 


<2php | 
include_once 'conn/conn.php'; // 载 入 数据 库 连 接 文 件 | 
$count = 1; | 
$sqlstr = 'select * from tb_object where id=' . $count; //sql 查询 语句 
S$rst = $conn->execute ( $sqlstr ) or die ('error: '. $conn->errorMsg 0 ); /执行 查询 语句 | 
if (false != $rst) { // 如 果 有 查询 结果 | 
for($i= 0: $i < $rst->FieldCount 0: $i++) { // 循 环 输出 各 个 字段 | 
Sfields = $rst->FetchField ( $i ); /生成 字段 信息 对 象 | 
$type = $fields->type: | 
2 | 
<td bgcolor="#FFFFFF"> | 
<?php | 
/从 对 象 中 获取 字段 的 类 型 信息 | 
if ($rst->metaType ( $type, - 1, $fields ) 一 "T) // 如 果 标准 类 型 为 “T” 
echo $conn->DBDate ( Srst->fields [$i] ): // 使 用 DBDate 函数 格式 化 时 间 | 
else 
echo Srst>fields [$i]: /如 果 是 其 他 类 型 ， 则 直接 输出 | 
3 | 
</td> | 
<?php 
! 
1 | 
Te 


运行 结果 如 图 9.11 所 示 。 
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专家 点 评 

在 本 实例 中 只 是 通过 FieldCount0 函 数 和 FetchField0) 函 数 对 数据 表 中 的 字段 进行 操作 。 在 
后 面 的 问题 中 将 应 用 EOF、 fields 和 movenextO 函 数 以 及 while 语句 完成 数据 表 中 数据 的 循环 输 
出 ， 其 实现 的 是 结果 集中 行 的 操作 。 


问题 202 ADODB 中 如 何 控 制 结果 集 的 存 取 方法 ? 


， 问题 阅 述 


| 专家 解答 


在 ADODB 中 如 何 控制 结果 集 的 存 取 方 法 ? 


在 通过 对 SQL 语句 的 操作 之 后 ， 多 数 都 会 返回 一 个 结果 集 (ADORecordSet 对 象 )。 通 过 


对 ADORecordSet 对 象 的 控制 ， 可 以 对 结果 集 进行 各 项 操作 ， 这 里 介绍 的 是 如 何 控制 结果 集 的 
| 存 取 方法 。 


1. SADODB COUNTRECS 
当 该 变量 为 tue 时 ，RecordCount() 函 数 会 在 数据 库 驱 动 不 支 持 select 指令 返回 的 记录 总 数 


时 进行 自动 模拟 , 返回 记录 数 , 默认 值 为 rue。 每 次 进行 查询 操作 时 都 会 自动 检查 该 变量 的 值 。 
| 由 于 这 个 功能 十 分 消耗 内 存 ， 所 以 不 建议 使 用 。 


2. $ADODB CACHE DIR 
该 变量 设置 缓存 目录 ， 一 般 在 使 用 缓存 函数 时 会 用 到 该 变量 ， 如 CacheExecute() 函 数 。 
3. $ADODB FETCH MODE 
该 变量 决定 结果 集 以 哪 种 方式 进行 存 取 ， 其 包括 4 个 值 : 
(1) define(ADODB FETCH DEFAULT',0): 
(2) define(ADODB FETCH NUM.'1) 
结果 集 按照 字段 序号 为 索引 进行 存 取 ， 如 Srst -> fields[0]、Srst -> fields[1]。 
(3) define(ADODB FETCH ASSOC'?) 
结果 集 按照 字段 名 称 为 索引 进行 存 取 ， 如 S$rst -> fields[bigclass]、Srst -> fields[id]。 
(4) define(ADODB FETCH BOTH'3) 
结果 集 同 时 支持 第 (2)、 第 (3) 两 种 方式 。 如 果 未 设置 该 变量 ， 则 默认 值 为 《1)， 不 同 


类 型 的 数据 库 驱 动 ， 默认 值 也 不 相同 。 这 里 不 建议 使 用 变量 (4), 因为 有 很 多 的 驱动 并 不 支持 。 


4. 应 用 示例 
在 本 示例 中 应 用 EOF、 fields 和 movenext0 函 数 以 及 while 语句 完成 数据 表 中 数据 的 循环 输 


出 ， 并 且 应 用 $ADODB_FETCH_MODE 控制 结果 集 的 存储 方式 。 其 具体 步骤 如 下 : 


(1) 创建 conn 文件 夹 ， 编 写 conn.php 文件 ， 载 入 adodb.iniphp 文件 。 实 现 与 MySQL 数 


| 据 库 服务 器 中 db_database 数据 库 的 连接 ， 应 用 SADODB_FETCH_MODE 设置 结果 集 以 字段 名 


称 为 索引 进行 存 取 ， 并 设置 数据 库 的 编码 为 UIF-8。 其 代码 如 下 : 
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<?php 

include_once ('../adodb5/adodb.inc.php'): // 载 入 adodb.ini.php 文件 

$conn =ADONewConnection(mysql); /连接 MySQL 数据 库 

$conn -> PConnect('localhost','root’,'l11','db_database’): // 连 接 db_database 库 

$ADODB FETCH MODE = ADODB FETCH ASSOC: /设置 结果 集 以 字段 名 称 为 索引 
$conn -> execute('set names utf8"); // 设 置 编 码 


> 


(2) 创建 index.php 文件 ， 完 成 数据 的 循环 输出 。 首 先 ， 包 含 数据 库 连 接 文件 。 然 后 ， 定 
义 SQL 查询 语句 ， 通 过 execute() 函 数 执行 查询 操作 。 最 后 ，while 循环 输出 查询 结果 。 其 关键 | 
代码 如 下 : | 


<?php 
include_once 'conn/conn php': // 载 入 数据 库 连 接 文件 
$sqlstr = 'select * from tb_user where id limit 5'; //sql 查询 语句 
S$rst = $conn->execute ( $sqlstr ) or die ( 'error: '. $conn->errorMsg () ); /执行 查询 语句 
while ( ! $rst->EOF ) { //wihle 语句 循环 输出 结果 
?> 
<tr> 
<td width="126" height="30" align="center"><?php echo $rst->fields [user]; ?></td> 
<td width="117" align="center"><?php echo S$rst->fields ['dates']; ?></td> 
</tr> 
<2?php 
$rst->movenext (); /指针 下 移 
} 
S$rst->close 0; // 关 闭 连 接 


$conn->close (); 
?> 


运行 结果 如 图 9.12 所 示 。 


图 9.12 ADODB 类 库 中 对 数据 库 中 数据 输出 的 控制 
专家 点 评 


在 专家 解答 中 已 经 对 ADODB 类 库 中 结果 集 的 存 取 方 式 进行 了 详细 介绍 ， 至 于 如 何 选择 ， | 
需要 根据 实际 情况 具体 问题 具体 分 析 ， 也 可 以 不 设置 结果 集 的 存 取 方式 。 建 议 读者 使 用 字段 名 | 
称 为 索引 进行 数据 存 取 ， 这 样 在 获取 数据 值 时 更 准确 一 些 ， 不 会 因为 数据 表 中 字段 的 添加 或 出 | 
除 而 影响 到 获取 的 值 。 | 
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问题 203 ”如 何 一 次 连接 两 个 数据 库 ? 


问题 前 述 
在 实际 的 程序 开发 中 ， 如 果 要 实现 数据 在 不 同 数据 库 服务 器 之 间 的 转换 ， 最 理想 的 方法 就 
是 编写 一 个 数据 转换 的 程序 ， 那 么 如 何 编写 该 程序 呢 ? 


专家 解答 


一 次 连接 两 个 数据 库 ， 即 同时 执行 两 次 ADONewConnection() 函 数 和 PConnect0 函 数 ， 分 
别 指定 不 同 MySQL 数据 库 服务 器 和 Access 数据 库 服 务 器 。 

在 连接 MySQL 服务 器 时 , 设置 的 用 户 名 是 root, 密码 是 111, 指定 的 数据 库 是 db_database， 
设置 的 编码 格式 是 UTF-8。 

在 连接 Access 服务 器 时 , 指定 的 数据 库 是 db_database mdb, 注意 这 里 设置 的 必须 是 Access 
数据 库 文件 在 服务 器 中 的 绝对 路 径 。 

同时 连接 两 个 数据 库 的 代码 如 下 : 

<?php 


include_once('..Jadodbs/adodb.inc.php): // 载 入 (Cinclude) adodb.inc php 文件 
$conn =ADONewConnection(mysql): /建立 连接 

$conn -> PConnect('localhost','root,'111','db_database'); /连接 数据 库 

$conn -> execute('set names utf8"): /设置 编码 格式 

$con = ADONewConnection("access"):; /| 连接 Access 数据 库 


$con-> PConnect("Driver={Microsoft Access Driver (*.mdb)}:;Dbq=F:\AppServ\www\mr\13\008\ 
conn\data\db_database.mdb"); 
$con-> execute('set names gb2312"): // 设 置 编 码 格式 
> 
在 将 从 MySQL 数据 库 中 读 取 的 数据 添加 到 Access 数据 库 时 , 应 用 的 是 ADODB 类 库 中 的 
GetInsertSQL() 函 数 。 使 用 该 函数 可 向 数据 库 中 添加 新 的 记录 ， 其 语法 如 下 : 
GetInsertSQL(&$rs, $arrFields) 
参数 $rs 设置 要 添加 记录 的 结果 集 ; 参数 $arrFields 设置 新 记录 的 字段 值 及 字段 名 称 。 
应 用 示例 
本 示例 同时 连接 两 个 不 同 的 数据 库 服务 器 , 完成 数据 由 MySQL 数据 库 到 Access 数据 库 的 
转换 。 其 具体 步骤 如 下 : 

(1) 创建 conn 文件 ， 编 写 conn.php 文件 ， 完 成 与 两 个 数据 库 服 务 器 的 连接 ， 其 方法 可 参 
考 专家 解答 。 

(2) 创建 index.php 文件 ， 将 从 MySQL 数据 库 中 读 取出 的 数据 通过 GetInsertSQLO 函 数 
添加 到 Access 数据 库 中 , 并且 在 本 页 中 输出 MySQL 数据 库 中 的 数据 ， 同 时 创建 数据 导出 和 查 
看 Aceess 数据 库 中 数据 的 超 链 接 。 其 关键 代码 如 下 : 

<2?php 


include ("conn/conn.php"); 
$sql = "select * from tb_user where id limit 5 "; /创建 sql 语句 
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Srst = $conn->execute ( $sql ) or die ('error: '. Sconn->errorMsg 0 ); 1/ 执行 查询 语 名 | 
Sarray = array (); | 
while (! Srst->EOF ) { | 
S$array [id] = $rst->fields ['id']; # process $arr | 
S$array [user] = iconv ( "utf-8", "gb2312", $rst->fields [user] ); # process $arr | 
Sarray ['pass'] = $rst->fields ['pass']; # process $arr | 


S$array ['dates'] = $rst->fields ['dates']; # process $arr | 
ifG_GET [inser] 一 "导出 { 
$sqls = "select * from tb_user where id=0 "; // 创 建 sql 语句 


$ret = $con->execute ( $sqls ) or die ('error: '. $con->errorMsg 0 ); /执行 查询 语句 | 

Sinsert = $con->getInsertSQL ( $ret, $array ) or die ( "update error: '. $con->errorMsg | 

0);W 添 加 新 数据 | 
$con->execute ( $insert ); | 

} | 

<t> | 
<td><?php echo Srst>fields [id]: ?2></td> | 

<td height="20"><2php echo Srst->fields ['user]: ?></td> | 

<td><?php echo $rst->fields ['dates']: ?></td> | 


(3) 创建 access.php 文件 ， 包 含 数据 库 连接 文件 conn.php， 完 成 Access 数据 库 中 数据 的 

循环 输出 。 

运行 本 示例 ， 单 击 图 9.13 中 的 “导出 ”按钮 ， 将 MySQL 数据 库 中 的 数据 导入 Access 数据 
库 中 ， 单 击 “ 查 看 ACCESS” 超 链接 将 看 到 如 图 9.14 所 示 的 存储 于 Access 数据 库 中 的 数据 。 


</tr> 
<2php 
S$rst->movenext (); /指针 下 移 | 
} 
S$rst->close |; // 关 闭 连 接 | 
$conn->close (); | 
?> 
| 
| 
| 


时 间 
2010.0+02 0c0c00 
20100402000000 
2326100403 0C0G00 
20100402 0c0G00 
2010.04.03 000000 


这 入 让 明日 科 技 有 了 皮 司 天 权 所 有 


图 9.13 ”MySQL 数据 库 中 的 数据 图 9.14 ”Access 数据 库 中 的 数据 


通过 ADODB 连接 Access 数据 库 时 ， 在 设置 数据 库 文件 的 存储 位 置 时 ， 必 须 指定 数据 库 
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文件 在 服务 器 中 的 绝对 路 径 ， 同 时 为 了 确保 Access 数据 库 文件 的 安全 ， 可 以 为 Access 数据 库 
设置 密码 。 通 过 选择 Access 菜单 栏 中 的 “工具 ”一 “安全 ”一 “设置 数据 库 密 码 ” 命 令 即 可 
设置 Access 数据 库 的 密码 。 

若 想 撤销 数据 库 密码 ， 只 需 选 择 Access 菜单 栏 中 的 “工具 ”一 “安全 ”一 “撤销 数据 库 
密码 ”命令 ， 在 弹出 的 “撤销 数据 库 密 码 ” 对 话 框 中 输入 设置 时 的 密码 即 可 。 


问题 204 如 何在 ADODB 中 生成 HTML 表格 ? 


| 问题 阐述 


ADODB 如 何 生 成 HTML 表格 ? 


| 专家 解答 


在 ADODB 类 库 中 ， 有 一 种 方法 可 以 返回 表格 式 的 结果 集 ， 即 将 查询 结果 直接 以 表格 的 形 


式 进行 输出 。 这 种 方法 就 是 rs2html0 函 数 , 其 返回 一 个 HIML 表格 格式 的 结果 集 。 其 语法 如 下 : 


function rs2html($adorecordset,[$tableheader _ attributes], [$col titles]) 

参数 说 明 : 
@ $adorecordset: 要 返回 的 结果 集 。 
@ $tableheader attributes: 对 表格 的 参数 及 属性 设置 。 
日 $col titles: 对 字段 的 重新 命名 。 
要 运用 本 函数 ， 必 须 载 入 adodb 类 库 中 的 tohtmlinc.php 文件 。 
应 用 示例 

(1) 创建 conn 文件 夹 ， 编 写 conn.php 文件 ， 完 成 与 MySQL 数据 库 的 连接 。 

(2) 创建 index.php 文件 ， 包 含 数据 库 连 接 文件 ， 定 义 SQL 查询 语句 ， 应 用 rs2html() 函 


| 数 将 查询 结果 以 表格 的 形式 输出 。 其 关键 代码 如 下 : 


<2?php 
include_once 'conn/conn php': // 载 入 数据 库 连接 文件 
include_once '../adodb5/tohtml.inc.php'; // 载 入 tohtmlinc .php 文件 
$Ist = $conn -> execute('select * from tb_bccd"): // 返 回 查 询 结果 集 


rs2html($rst,' width="775" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" 


| bgcolor="#CCCCCC",array( 序 号 名称", 类 别 "' 时 间 )); 


?> 
专家 点 评 
Is2html() 函 数 是 一 个 独立 的 函数 (rs2heml = recordset to html), 相当 于 PHP 中 的 odbc_result_ 
all 函数 。 它 输出 一 个 完整 的 adorecordSet，$adorecordset 如 同一 个 HTML 表格 。S$tableheader 
attributes 允许 控制 表格 中 的 参数 ， 如 cellpadding、cellspacing 或 border 等 属性 。 最 后 ， 通 过 
$col titles 参数 设置 数据 表 中 字段 的 名 称 。 注 意 ， 它 只 是 一 个 快速 除 错 的 机 制 ， 并 不 是 一 个 理 
想 的 结果 集 输出 方法 。 
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问题 205 如 何在 ADODB 中 生成 下 拉 列 表 框 ? 


问题 阐述 
ADODB 如 何 生成 下 拉 列 表 框 ? 
专家 解答 
ADODB 中 生成 下 拉 列 表 框 应 用 的 是 GetMenu() 函 数 。 | 
GetMenu0 函 数 返回 一 个 HTML 下 拉 列 表 框 (select)。 结 果 集 的 第 一 列 (fields[0]) 被 显示 | 
到 下 拉 列 表 中 ; 第 二 列 (fields[1]) 被 设置 为 <option> 标 签 的 value 属性 。 其 语法 如 下 : | 
GetMenu($name, [$default str="], [$blank1stiteny-true], [Smultiple select-false] [Ssize=0], [SmoreAtt="D 

参数 说 明 : 

@ $name: 下 拉 列 表 框 的 名 称 。 | 
@ $default_str: 如 果 $default_str 的 值 等 于 其 中 一 个 fields[0]，, 那么 这 个 fields[0] 就 是 该 下 拉 | 
列表 的 默认 选项 ， 相 当 于 为 这 个 fileds[0] 添 加 了 属性 <selected>。 | 
日 Sblank: 该 参数 默认 值 为 tue。 在 生成 下 拉 列 表 框 时 ， 第 一 行 选 项 默认 为 空 ， 即 | 
<option></option>。 如 果 为 false， 则 没有 默认 的 空 选 项 。 | 
@ $multiple_select: 是 否 支持 多 选 。 默 认 值 为 false， 即 不 允许 。 | 
@ $size: 列表 框 选取 的 大 小 。 默 认为 0， 生 成 一 个 下 拉 列 表 。 如 果 $size 的 值 大 于 0， 则 生 | 
成 一 个 列表 框 。 | 
@ $moveAttr: 该 参数 可 以 增加 列表 框 的 属性 。 例 如 添加 一 个 JavaScript 脚本 等 。 
应 用 示例 | 
应 用 ADODB 中 的 GetMenu2() 函 数 将 查询 结果 集中 的 数据 直接 输出 到 下 拉 列 表 框 中 。 其 具 | 
体 步骤 如 下 : | 
(1) 创建 conn 文件 夹 ， 编 写 conn.php 文件， 完成 与 MySQL 数据 库 的 连接 。 | 
(2) 编写 index.php 文件 ， 创 建 form 表单 ， 设 置 action 属性 值 为 mndex_okphp。 包 含 数据 | 
库 连 接 文 件 ， 定 义 SQL 查询 语句 ， 应 用 GetMenu20 函 数 将 查询 结果 以 下 拉 列 表 框 的 形式 输出 。 | 
其 关键 代码 如 下 : | 


<?php 
include_once 'conn/conn php': // 载 入 数据 库 连 接 文件 
$sqlstr = 'select name,id from tb_bccd': /sql 查询 语句 


$rst = $conn -> execute($sqlstr) or die(execute error: '.$conn -> errorMse()); 
/执行 查询 语句 ， 返 回 结果 集 
> 
<!-- ”调用 getMenu20 函 数 ”--> 
<2php echo $rst -> getMenu?2('type',$_ POST[name'],false): ?> 


运行 结果 如 图 9.15 所 示 。 
(3) 创建 index_ok.php 文件 ， 获 取 表单 中 提交 的 数据 ， 完 成 用 户 的 登录 操作 。 
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图 9.15 ADODB 生成 下 拉 列 表 框 


| 专家 点 评 


不 知道 读者 是 否 注意 到 ， 在 本 示例 中 运用 的 并 不 是 专家 解答 中 讲解 的 GetMenu0 函 数 ， 而 


| 是 GetMenu20 函 数 ,GetMenu20 函 数 的 作用 与 GetMenu0 函 数 基本 相同 , 其 区 别 在 于 : GetMenu() 
| 中 的 参数 $default_str 和 fields[0] 比 较 ， 如 果 相等 ， 那 么 fields[0] 就 是 这 个 下 拉 列 表 的 默认 选项 ; 
| 而 getMenu20 中 的 参数 gdefault str 则 是 和 fields[1] 进 行 比较 ， 其 他 不 变 。 


问题 206 一 个 小 巧 的 分 页 函数 一 一 ADODB _Pager() 


| 问题 阐述 


在 ADODB 中 ， 有 一 种 简单 的 数据 分 页 输出 的 方法 ， 它 被 封装 到 一 个 函数 中 ， 这 就 是 


| ADODB_Pager0 函 数 。 那 么 这 个 函数 是 如 何 使 用 的 呢 ? 
| 专家 解答 


| 页 功能 。 


ADODB Pager0 函 数 是 ADODB _ Pager 类 的 构造 函数 ,通过 类 中 的 renderO 函 数 可 以 实现 分 


ADODB Pager0 函 数 的 语法 如 下 : 
ADODB Pager($conn, $sql, $id = 'adodb', $showPageLinks = false) 

参数 说 明 : 

@ $conn: 是 数据 库 连 接 对 象 ; 

@ $sql: 是 执行 的 SQL 语句 ; 

@ $id: 每 个 分 页 的 id 号 ; 

@ $showPageLinks: 设置 是 否 显示 各 个 页 的 链接 ， 默 认为 false。 

应 用 示例 

应 用 ADODB Pager0 函 数 实 现 一 个 分 页 功能 。 其 具体 步骤 如 下 : 
| (1) 创建 index.php 文件 ， 包 含 数据 库 连 接 文件 conn.php， 包 含 adodb5 中 的 
| adodb-pagerinc.php 文件 ,实例 化 ADODB Pager 类 ,传递 查询 参数 ,设置 每 页 显示 10 条 记录 。 
| 代码 如 下 : 


<?php 

include 'conn/conn php': // 载 入 数据 库 连 接 文件 
include_once '../adodb5/adodb-pager.inc.php'; // 载 入 adodb_parperiniphp 文件 
人 # ”声明 对 象 */ 


S$pager = new ADODB_Pager($conn,"select id,user as ' 用 户 名 ', pass as ' 密 码 ',dates as ' 注 册 时 间 ' 
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from tb_user"); 
人 # 调用 $Render 函数 确定 每 页 显示 多 少 */ 
$pager -> Render(10); 


ye 
(2) 打开 adodbs 文件 夹 下 的 adodb- HE 本 
pager.inc.php 文件 ， 重 新 设置 页 面 的 样式 。 2 
运行 结果 如 图 9.16 所 示 。 | 
1 Te 
应 用 ADODB_Pager0 函 数 实现 分 页 功 i 


能 有 一 个 前 提 条 件 ， 就 是 在 包含 ADODB 加 916 应 用 ADODB_Pager0 函 数 实现 一 个 分 页 功能 
中 的 adodb.inc.php 文件 的 同时 还 要 包含 


adodb-pagerinc.php 文件 。 
问题 207 如何 通 过 事务 处 理 机 制 保证 数据 库 
操作 的 完整 性 ? 
问题 讲述 


事务 处 理 机 制 在 程序 开发 过 程 中 有 着 非常 重要 的 作用 ， 它 可 以 使 整个 系统 更 加 安全 。 如 在 | 
银行 处 理 转账 业务 时 ， 如 果 A 账户 中 的 金额 刚 被 转 出 ， 而 B 账户 还 没 来 得 及 接收 就 发 生 停电 | 
事故 ， 这 会 给 银行 和 个 人 带 来 很 大 的 经 济 损失 。 采 用 事务 处 理 机 制 ， 一 旦 在 转账 过 程 中 发 生意 | 
外 ， 则 程序 将 回 滚 ， 不 做 任何 处 理 ， 那 么 在 ADODB 中 是 如 何 实现 事务 处 理 的 呢 ? | 


专家 解答 


在 ADODB 中 ， 事 务 处 理应 用 的 是 如 下 3 个 函数 。 | 
(1) BeginTrans0 函 数 : 开启 事务 处 理 。 该 函数 和 接 下 来 的 两 个 函数 都 是 应 用 在 PHP 的 事 | 
务 处 理 中 。 | 
(2) CommitTrans() 函 数 : 如 果 成 功 完成 数据 库 操 作 ， 则 执行 该 函数 。 | 
(3)》RollbackTrans0 函 数 ， 结 束 一 次 操作 ， 恢 复 操作 前 的 所 有 改变 。 如 果 成 功 则 返回 Tue， | 
如 果 服 务 器 不 支持 则 返回 False。 | 
应 用 示例 | 
本 示例 实现 数据 在 不 同 数据 库 间 的 转换 功能 , 在 将 MySQL 数据 库 中 数据 转换 到 Access 数 | 
据 库 后 ， 删 除 MySQL 数据 库 中 的 原始 数据 ， 并 应 用 事务 处 理 机 制 保证 数据 转换 操作 安全 、 顺 | 
利 地 进行 。 其 具体 实现 步骤 如 下 : | 
(1) 创建 conn 文件 夹 ， 编 写 conn php 文件 ， 完 成 与 MySQL 数据 库 的 连接 。 | 
(2) 首先 , 编写 index.php 文件 , 包含 数据 库 连接 文件 , 定义 SQL 语句 , 循环 输出 MySQL | 
数据 库 中 的 数据 ， 通 过 BeginTrans( 〇 函数 开启 事务 处 理 。 然 后 ， 根 据 超 链 接 传递 的 参数 进行 判 | 
断 , 通过 getInsertSQLO 函 数 执行 数据 的 转移 操作 ,转移 成 功 后 通过 delete 语句 删除 原始 数据 库 | 
中 的 数据 。 最后， 根据 添加 语句 和 删除 语句 的 执行 情况 ， 判 断 是 执行 CommitTrans 0 提交 操作 | 


X297 


人 Pp 六 须知 道 的 300 个 问题 


还 是 执行 RollbackTrans 0) 回 滚 操作 。 其 关键 代码 如 下 : 


| <?php 
| include ("conn/conn php"); 
| $sql = "select * from tb_user where id order by id limit 5 "; // 创 建 sql 语句 
食 二 | | S$rst = $conn->execute ( $sql ) or die ('error: '. $conn->errorMsg 0 ); /执行 查询 语句 
$array = array (); 
Senso (E // 开 始 事务 处 理 


while (! $rst->EOF ) { 
寺 ($_GET [insert] 一 "数据 转移 ") { 
S$array [id] = S$rst->fields ["id']; # process $arr 
$array [user] = iconv ( "utf-8", "gb2312", $rst->fields [user] ); # process $arr 
$array ['pass'] = Srst->fields [pass']; # process $arr 
$array ['dates'] = S$rst->fields ['dates']: # process $arr 
$sqls = "select * from tb_user where id=0 "; // 创 建 sql 语句 
$ret = $con->execute ( $sqls ) or die ( 'error: ' . $con->errorMsg 0 ); // 执 行 查询 语句 
S$insert = $con->getInsertSQL ( $ret, $array ) or die ( "update error:' . $con->errorMsg 0 ): 
/添加 新 数据 


TO /开始 事务 处 理 


$result=$con->execute ( $insert ); 
$sql ='delete from tb_user where id =' . $rst->fields [id]; 。 //sql 删除 语句 
$results = $conn->execute ( $sql ) or die ( 'execute error: '. $conn->ErrorMsg () ); 


/执行 删除 语句 
?> 
<tr> 
<td><?php echo $rst->fields [id]; ?></td> 
<td height="20"><?php echo S$rst->fields [user]; ?></td> 
<td><?php echo S$rst->fields ['dates']; ?></td> 
</tr> 
<2php 
S$rst->movenext (): /指针 下 移 
} 
if(Sresul){ 
$con->CommitTrans 0; // 执 行 提交 
if ($results) { 
$conn->CommitTrans (): /执行 提交 
| }else{ 
| $conn->RollbackTrans (): // 执 行 回 深 操 作 
| 3? 
| Jelse{ 
| $con->RollbackTrans |): // 执 行 回 滨 操 作 
| } 
| S$rst->close (): // 关 闭 连接 
| $conn->close 0: 


| 2> 
| (3) 创建 access.php 文件 ， 循 环 输出 Access 数据 库 中 存储 的 数据 。 

运行 本 示例 ， 如 图 9.17 所 示 ， 单 击 “ 数 据 转移 ” 超 链 接 ， 将 MySQL 数据 库 中 的 数据 转移 
到 Access 数据 库 中 ， 并 且 删 除 MySQL 数据 库 中 的 原始 数据 。 单 击 “ 查 看 ACCESS” 超 链接 ， 
将 查看 到 Access 数据 库 中 存储 的 数据 ， 如 图 9.18 所 示 。 


pp 2 298 


汗 9 填 要 据 订 宙 尼 一 


图 学 涅 在 线 
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图 9.17 数据 转移 图 9.18 查看 Access 数据 库 中 的 数据 
专家 点 评 


事务 是 一 个 或 一 系列 的 查询 ， 这 些 查 询 要 么 全 部 执行 ， 要 么 全 部 不 执行 。 作用 是 人 有 
器 发 生 错误 或 骨 溃 的 情况 下 确保 数据 库 的 一 致 性 。 

一 个 事务 被 永久 地 写 入 到 数据 库 中 称 事务 提交 ,将 状态 重 轩 到 事务 开始 之 前 的 状态 称 为 事 | 
务 回 滨 。 事 务 具 有 以 下 4 个 特性 。 

原子 性 : 指 事务 作为 一 个 整体 要 么 全 部 执行 ， 要 么 全 部 不 执行 。 
改 性 : 指 一 个 事务 必须 能 够 使 数据 处 于 一 致 的 状态 。 

孤立 性 :在 事务 完全 完成 之 前 ， 它 们 都 是 孤立 的 。 
持续 性 : 一 旦 写 入 数据 库 后 ， 事 务必 须 是 永久 的 、 持 续 的 。 


问题 208 ADODB 如 何 输 出 系统 中 执行 的 SQL 语句 ? 


a 
| 
洪 
谋 


问题 阐述 


ADODB 提供 很 多 操作 数据 库 的 方法 和 函数 , 这 其 中 不 乏 错误 处 理 机 制 和 程序 调试 的 方法 ， 
那么 如 何 输出 系统 中 执行 的 SQL 语句 呢 ? 


专家 解答 


输出 系统 中 执行 的 SQL 语句 ,应 用 的 是 ADODB 中 的 debug 变量 ,如 果 变量 被 设 定 为 Tue，| 
那么 当 有 输出 操作 时 ， 同 时 也 输出 调试 信息 ， 如 果 启 用 debug 变量 ， 那 么 当 错 误 发 生 时 将 自动 
调用 ErrorMsg() 函 数 返回 错误 信息 。 | 
ErrorMsg0 函 数 返 回 最 后 的 状态 或 出 错 信 息 ， 即 使 没有 错误 发 生 ， 也 会 返回 一 个 字 串 。 所 ， 
一 般 情况 下 ， 只 有 在 发 生 错误 时 (返回 False》 才 调用 该 函数 。 | 
应 用 示例 


在 本 示例 中 设置 debug 变量 的 值 为 Tue， 开 启 调试 功能 ， 输 出 程序 中 执行 的 SQL 语句 ， 
如 果 SQL 语句 编写 错误 ， 那 么 将 输出 错误 信息 。 其 具体 步骤 如 下 : 

(1) 创建 conn 文件 夹 ， 编 写 conn php 文件 ， 完 成 与 MySQL 数据 库 的 连接 。 | 

(2) 编写 index.php 文件 。 首 先 ， 包 含 数 据 库 连 接 文 件 ， 设 置 debug 变量 的 值 为 True， 开 | 
启 调 试 功 能 。 然后， 定义 SQL 语句 ， 循 环 输出 MySQL 数据 库 中 的 数据 。 最后， 关闭 连接 、 关 | 
闭 数据 库 。 其 关键 代码 如 下 : 

<?php 
include ("conn/conn php"); 
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$conn -> debug = true; /开启 调试 
$sql = "select* from tb_use where id limit 5 "; /创建 sql 语句 
$ret = $conn->execute ( $sql ) : // 执 行 查询 语句 
while ( ! $ret->EOF ) { /循环 输出 


| <td height="20"><?php echo Sret>fields[0]; ?></td> 
<td height="20"><?php echo $ret->fields[1]; ?></td> 
<td><?php echo $ret->fields[3]; ?></td> 


</tr> 
<?php 
S$ret->movenext (); // 指 针 下 移 
} 
$ret->close (): // 关 闭 连 接 
$conn->close (); 
> 


运行 结果 如 图 9.19 所 示 。 


ww MR LS OLDinder php oa ine 


图 9.19 输出 SQL 语句 中 的 错误 信息 


| 专家 点 评 

| Debug 变量 的 调试 功能 适合 在 程序 开发 过 程 中 应 用 , 通过 它 可 以 及 时 发 现 SQL 语句 中 的 错 
| 误 。 但 是 在 实际 的 应 用 环境 中 ， 就 必须 关闭 这 个 调试 功能 ， 为 了 确保 程序 数据 库 的 安全 ， 如 果 
| 存在 SQL 语句 的 错误 ， 还 是 不 要 让 浏览 者 看 到 为 好 ， 否 则 不 但 会 影响 到 浏览 者 访问 网 站 的 心 
| 情 ， 更 重要 的 是 容易 暴露 网 站 数据 库 的 信息 。 


问题 209 ADODB 如 何 应 用 errorMsg() 丁 数 返回 错误 信息 ? 


| 问题 阐述 
| ADODB 如 何 应 用 errorMsg0 函 数 返回 错误 信息 ? 
| 专家 解答 


| errorMsg0 函 数 用 于 返回 最 后 的 状态 或 出 错 信 息 ， 即 使 没有 错误 发 生 ， 也 会 返回 一 个 字 串 。 

| 所 以 ， 它 多 数 情况 下 是 与 die0 函 数 联合 使 用 的 ， 在 执行 的 SQL 语句 发 生 错 误 时 (返回 False) 
| 才 调 用 该 函数 。 
| 应 用 示例 
| 本 示例 应 用 die0 函 数 在 执行 的 SQL 语句 发 生 错误 时 ， 调 用 errorMsg() 函 数 返 回 错误 信息 。 
| 其 具体 步骤 如 下 : 

(1) 创建 conn 文件 夹 ， 编 写 conn php 文件 ， 完 成 与 MySQL 数据 库 的 连接 。 

(2) 编写 indexphp 文件。 首先 ， 包 含 数据 库 连接 文件 。 然 后 ， 定 义 SQL 语句 ， 通 过 execute0) 

了 . 300 . 


9 CAS 
第 日 章 数据 库 柚 训导 和 


函数 执行 SQL 语句 ， 应 用 die0 函 数 在 SQL 语句 的 执行 发 生 错误 时 ， 调 用 crorMse0 机 数 返 回 | 
戎 误 信 息 。 最 后 ， 循 环 输出 MySQL 数据 库 中 的 数据 。 其 关键 代码 如 下 : | 


<?php 
include ("conn/conn php"); | 
$sql = "select * from tb_use where id limit 5 "; // 创 建 sql 语句 | 策 - 
S$ret= $conn -> execute($sql) or die($conn -> errorMsg0):// 执 行 查询 语句 ， 并 返回 错误 信息 ， 
while ( ! $ret->EOF ) { /循环 输出 
?> 
<tr> 


<td height="20"><?php echo $ret->fields[0]; ?></td> 
<td height="20"><?php echo S$ret->fields[1]; ?></td> 
<td><?php echo $ret->fields[3]; ?></td> 


</tr> 
<?php 
$ret->movenext (); /指针 下 移 
$ret->close 0: /关闭 连接 
$conn->close (); 
?> 
运行 结果 如 图 9.20 所 示 。 Errorgst 返 回 错 误 信息 
专 家 点 评 wy hoy tb_use oe 


errorMsg0 函 数 返回 最 后 的 状态 或 出 错 信息 ， 其 特点 是 无 ee 
论 SQL 语句 是 耕 出错 它 者 会 返回 一 个 字符 串 ， 这 就 要 求 在 应 。 。 图 920 返回 错误 信息 
用 errorMsg0 函 数 时 必须 注意 .为 避免 在 SQL 语句 正确 的 情况 
下 返回 字符 串 ， 所 以 errorMsg0 函数 在 SQL 语句 的 执行 发 生 错 误 时 运用 是 最 佳 方案 。 


问题 210 ADODB 如 何 返 回 所 有 错误 信息 ? 


问题 阐述 
在 ADODB 中 如 何 返 回 所 有 错误 信息 ? 
专家 解答 | 
ADODB 中 提供 的 这 个 自 定义 的 错误 处 理 机 制 存储 于 adodb-errorhandler.inc.php 文件 中 ， 通 | 
过 它 可 以 调用 PHP 函数 error reporting 控制 错误 信息 的 输出 样式 。 | 


应 用 ADODB 中 提供 的 自 定义 错误 处 理 机 制 ， 必 须 在 程序 连接 数据 库 服 务 器 之 前 载 入 | 
adodb-errorhandler.inc.php 文件 。 | 
应 用 示例 | 
本 示例 运用 ADODB 类 库 中 提供 的 自 定义 错误 处 理 机 制 , 调用 PHP 函数 error reporting 设 | 
置 错 误 级 别 为 E ALL， 显 示 所 有 的 错误 信息 。 其 具体 实现 步骤 如 下 : | 
(1) 创建 comn 文件 夹 , 编写 conn php 文件 。 首 先 ， 设 置 error_ reporting0) 函 数 的 值 为 E ALL， | 
301. ey 
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| 库 的 连接 。 其 代码 如 下 : 


全 A 


| 显示 所 有 的 错误 信息 。 然 后 ， 载 入 adodb-errorhandlerinc.php 文件 。 最后， 完成 与 MySQL 数据 


<2php 
error_reporting(E_ALL): // 显 示 所 有 的 错误 信息 
include_once ('../adodb5/adodb-errorhandler.inc.php"); 
include_once ('../adodb5/adodb.inc.php"); // 载 入 adodb.ini.php 文件 
S$conn = ADONewConnection('mysql); // 连 接 MySQL 数据 库 
$conn -> PConnect('localhost','root','111','db_database’); // 连 接 db_database 库 
/ADODB FETCH MODE =ADODB FETCH ASSOC; // 设 置 结果 集 以 字段 名 称 为 索引 
S$conn -> execute('set names utf8"); // 设 置 编码 
re 
(2) 编写 index.php 文件 。 首先, 包含 数据 库 连 接 文 件 。 然后 ,定义 SQL 语句 , 通过 executeO 
函数 执行 SQL 语句 。 最 后 ， 循 环 输出 MySQL 数据 库 中 的 数据 。 其 关键 代码 如 下 : 
<?php 
include ("conn/conn.php"); 
$sql = "select * from tb use where id limit 5 "; /创建 sql 语句 
$ret = $conn -> execute($sq]l); // 执 行 查询 语句 
while ( ! $ret->EOF ) { // 循 环 输出 
?> 
<tr> 
<td height="20"><?php echo $ret->fields[0]: ?></td> 
<td height="20"><?php echo $ret->fields[1]; ?></td> 
<td><?php echo $ret->fields[3]; ?></td> 
</tr> 
<2php 
$ret->movenext 0); // 指 针 下 移 
} 
$ret->close 0: // 关 闭 连接 
$conn->close (); 
?> 
运行 结果 如 图 9.21 所 示 。 Error_repertint 返 回 所 有 铺 误 信息 
专家 点 评 


在 本 示例 中 ， 设 置 error_reporting() 函 数 的 值 为 E_ALL 返 
回 所 有 的 错误 信息 。 如 果 将 值 设置 为 0， 则 不 会 有 任何 错误 信 


息 输出 。 


问题 211 


问题 讲述 


通常 只 是 应 用 


将 错误 信息 存储 于 日 志文 件 中 呢 ? 


图 9.21 返回 所 有 错误 信息 


如 何 将 错误 信息 存储 于 日 志文 件 中 ? 


ADODB 类 库 中 提供 的 自 定义 错误 处 理 机 制 返回 所 有 的 错误 信息 , 那么 如 何 
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专家 解答 

要 想 将 错误 信息 存储 于 日 志文 件 中 , 应 用 ADODB 中 提供 的 自 定义 错误 处 理 机 制 , 必须 在 | 
程序 连接 数据 库 服务 器 之 前 载 入 adodb-errorhandlerincphp 文件 。 | 

将 错误 信息 存储 于 指定 的 文件 中 ， 应 用 的 是 两 个 选择 性 常量 : ADODB _ ERROR LOG 
TYPE 和 ADODB ERROR LOG DEST。 

ADODB_ERROR LOG TYPE 常量 设置 错误 记录 的 类 型 ， 可 选 值 如 下 : 

回 0: 默认 值 ， 根 据 在 php.ini 文件 中 的 error log 配置 ， 错 误 被 发 送 到 服务 器 日 志 系 | 

统 或 文件 。 | 

1: 错误 被 发 送 到 destination 参数 中 的 地 址 。 只 有 该 类 型 使 用 headers 参数 。 

2: 通过 PHP debugging 连接 来 发 送 错误 。 该 选项 只 在 PHP 3 中 可 用 。 

回 3: 错误 发 送 到 文件 目标 字符 串 。 

将 ADODB_ ERROR LOG TYPE 的 值 设置 为 3， 即 将 错误 信息 存储 于 常量 ADODB_ 
ERROR_ LOG_DEST 指定 的 文件 中 。 | 

例如 ， 指 定 ADODB_ERROR LOG _DEST 的 值 为 erorerrorlog， 即 将 错误 信息 存储 于 项 | 
目 根 目录 下 error 文件 夹 的 errorlog 文件 中 。 | 

应 用 示例 | 

在 本 示例 中 , 运用 ADODB 中 的 错误 处 理 机 制 ， 不 但 返回 错误 信息 ， 而 且 将 错误 信息 存储 | 
到 指定 的 文件 中 。 其 具体 步骤 如 下 : | 

(1) 创建 conn 文件 夹 ， 编 写 conn.php 文件 。 首 先 ， 定 义 两 个 常量 设置 程序 中 错误 信息 的 | 
存储 位 置 。 然 后 ， 载 入 adodb-errorhandlerinc.php 文件 。 最 后 ， 完 成 与 i 数据 库 的 连接 。 | 
其 代码 如 下 : 


<2php 

define( ADODB ERROR LOG TYPE',3); /设置 类 型 

define( ADODB ERROR LOG DEST',error/error.log"): // 设 置 存 储 路 径 

include_once ('../adodb5/adodb-errorhandler.inc.php'); // 载 入 自 定义 错误 处 理 文件 
include_once ('../adodb5/adodb.inc.php"):; // 载 入 adodb.ini.php 文件 

$conn = ADONewConnection(mysql); /连接 MySQL 数据 库 | 
$conn -> PConnect('localhost','root','l11','db_database'): // 连 接 db_database 库 | 
$conn -> execute('set names utf8"): // 设 置 编 码 | 
2 | 


(2) 编写 index.php 文件 。 首 先 ， 包 含 数据 库 连 接 文件 。 然 后 ， 定 义 SQL 语句 ， 通 过 
execute() 函 数 执行 SQL 语句 。 最 后 ， 循 环 输出 MySQL 数据 库 中 的 数据 。 其 关键 代码 如 下 : 


<?php 

include ("conn/conn.php"); 
$sql = "select * from tb_use where id limit 5 "; /创建 sql 语句 | 
$ret = $conn -> execute($sql): // 执 行 查询 语句 | 
while (! $ret->EOF ) { // 循 环 输出 | 
人 | 
<t> | 
<td height="20"><?php echo S$ret->fields[0]; ?></td> | 
<td height="20"><?php echo $ret->fields[1]: ?></td> | 
<td><?php echo S$ret->fields[3]:; ?></td> | 
</tr> | 
<?php ! 
*303。 : 
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} 


Sret->movenext (); /指针 下 移 
| $ret->close (0): /关闭 连接 
| $conn->close (); 
天 六 | > 
| rr 
| 在 本 示例 中 将 返回 的 错误 信息 存储 于 根 目录 下 EE 习 
的 error 文件 夹 中 ，errorlog 中 的 错误 信息 如 图 9.22 | 


| 所 示 。 图 9.22 通过 日 志文 件 存储 错误 信息 
问题 212 ”如 何 封装 ADODB 操作 数据 库 类 ? 


问题 阐述 


| 既然 可 以 将 ADODB 连接 数据 库 的 方法 封装 到 类 中 , 那么 如 何 将 ADODB 操作 数据 库 的 方 
| 法 封装 到 类 中 ? 
| 专家 解答 
| ADpoDB 操作 数据 库 类 是 AdminDB, 在 操作 数据 库 的 ExecSQL( 方 法 中 关键 是 如 下 几 个 函 
| 数 的 运用 。 
(1) strtolower() 函 数 ， 将 字符 串 转换 为 小 写字 母 。 其 语法 如 下 : 
string strtolower(string str) 
运用 本 函数 的 目的 是 将 SQL 语句 中 的 字符 串 都 转换 为 小 写 。 
(2) substr0 函 数 ， 从 指定 的 字符 串 中 按照 指定 的 位 置 截取 一 定 长 度 的 字符 。 其 语法 如 下 : 
string substr(string str,int start,int length) 
参数 说 明 : 
@ str: 指定 本 截取 的 字符 串 。 
@ start: 指定 开始 截取 字符 串 的 位 置 。 如 果 参 数 start 为 负数 , 则 从 字符 串 的 末尾 开始 截取 。 
| 日 length: 可 选 参数 。 指 定 截取 字符 串 的 长 度 。 如 果 length 为 负数 ， 则 表示 取 倒 数 第 length 
| 个 字符 。 


| 技巧: 
| 本 函数 中 参数 start 的 指定 位 置 是 从 0 开始 计算 的 ， 即 字符 囊 中 的 第 一 个 字符 表示 为 0。 


运用 substr0 函 数 对 SQL 语句 中 前 6 个 字符 串 进行 截取 ， 其 目的 是 获取 SQL 语句 的 类 型 
(insert、select 或 update)， 判 断 其 执行 的 是 什么 操作 。 
(3) trim0 函 数 ， 删 除 字符 串 中 首尾 的 空白 或 其 他 字符 。 
| (4) execute0 函 数 ，ADODB 中 执行 SQL 语句 的 方法 ， 并 返回 一 个 结果 集 (ADORecordSet 
| 对 象 )， 失 败 则 返回 False。 其 语法 如 下 : 
| execute($sql[,Sinputar—false]) 


参数 说 明 : 
@ $sql: 指定 要 执行 的 SQL 语句 。 
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@ $inputarr: 设置 传 入 的 结合 变量 ， 如 果 没 有 设置 Sinputarr， 则 $sql 为 普通 的 SQL 语句 ， 
那么 execute 的 格式 如 下 : 
$connect->execute('select * from tb_user where id= 1") 
如 果 设 置 了 $inputarr， 那 么 execute 的 格式 如 下 : 
$connect -> execute('select * from tb_user where id = ?',array($v])) 

(5) GetRows() 函 数 是 GetArray() 的 同 义 函数 ， 是 为 了 与 Microsoft ADO 相 容 才 有 的 。 其 | 
语法 如 下 : | 
GetRows ([$number_of rows]) | 
返回 从 当前 指针 指向 的 记录 开始 ， 到 ($number_of rows - 1) 行 的 全 部 记录 的 数组 。 参 数 | 
$number_of rows 是 指定 的 记录 行 。 如 果 没 有 给 出 ， 则 一 直到 EOF 才 停 止 。 | 

应 用 示例 | 
封装 数据 库 操作 类 AdminDB， 同 时 应 用 数据 库 连 接 类 ConnDB 实现 与 MySQL 数据 库 | 
db_database 的 连接 ， 应 用 AdminDB 操作 MySQL 数据 库 ， 循 环 读 取 数据 库 中 的 数据 。 其 具体 | 
步骤 如 下 : | 


(1) 创建 found.database.php 文件 ， 定 义 数据 库 操作 类 AdminDB， 定 义 ExecSQL( 方 法 ， | 
根据 传递 的 SQL 语句 执行 对 数据 库 的 操作 。 在 ExecSQL( 方 法 中 ， 应 用 substr0 函 数 获取 参数 | 
传递 的 SQL 语句 的 类 型 ， 然 后 根据 SQL 语句 的 类 型 进行 判断 ， 如 果 是 select 查询 语句 ， 则 执 | 
行 GetRow() 方 法 , 获取 查询 结果 ; 如 果 是 update、insert 或 delete 语句 , 则 直接 返回 True 或 False。 | 
其 代码 如 下 : 

class AdminDB{ 
function ExecSQL($sqlstr,$conn){ /定义 方法 ， 参 数 为 SQL 语句 和 连接 数据 库 返 回 的 对 象 
$sqltypestrtolower(substr(tim($sqlstD,0.0)/ 和 截取 SQL 中 的 前 6 个 字符 串 ， 并 转换 成 小 写 | 


S$rs=$conn->Execute($sqlstr); // 执 行 SQL 语句 
if($sqltype—"select"){ // 判 断 如 果 SQL 语句 的 类 型 为 SELECT 
Sarray=$rs->GetRows(): // 执 行 该 语句 ， 获 取 查 询 结 果 
这 count($array) 一 0 上 | $rs 一 false) ”“”// 判 断 语 句 是 否 执 行 成 功 | 
Tetum false: /如果 查询 结果 为 0 或 执行 失败 ， 则 返回 False | 
else | 
return $array: // 否 则 返回 查询 结果 的 数组 


}elseif ($sqltype—"update" || $sqltype 一 "insert" || $sqltype—"delete"){ 
// 判 断 如 果 SQL 语句 类 型 不 为 SELECT 则 执行 如 下 操作 


if($rs) 

Tetum true; // 执 行 成 功 返 回 True 
else 

return false: // 否 则 返回 False 


上 
} 
) 


(2) 创建 connect.database.php 文件 ， 实 例 化 数据 库 连 接 类 ConnDB 和 数据 库 红 作 关 | 
AdminDB。 其 关键 代码 如 下 : 


<?php 
Tequire("found.database php"): /包含 类 文件 
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$connobj=new ConnDB("mysql","localhost","root","111","db_database",false); /数据 库 连 接 类 实例 化 


$conmn=$connobj->GetConnIdO: // 返 回 连 接 标 识 
$admindb=new AdminDB0: /| 数据 库 操作 类 实例 化 
| 2> 
公 7】 | G3) 创建 ndexphp 文件 ， 包 含 数据 库 连接 类 和 数据 库 操作 类 的 实例 化 文件 ， 定 义 SQL 


语句， 调用 数据 库 操作 类 中 的 _ExeeSQL0 方 法 执行 SQL 查询 语句 ， 最 后 通过 for 语句 循环 
ExecSQL0 方 法 返回 的 查询 结果 。 其 关键 代码 如 下 : 


<2php 
include_once 'conn/connect.database.php'; // 包 含 数据 库 连接 和 操作 类 的 实例 化 文件 
$sqlstr = 'select * from tb_bccd' // 定 义 SQL 语句 
$array = $admindb->ExecSQL($sqlstr, $conn); // 调 用 数据 库 操作 类 中 的 方法 ， 执 行 查询 操作 
for($i=0;$i<count($array);$i++){ // 通 过 for 循环 输出 数据 库 中 的 数据 
?> 

<tr> 


<td align="center" bgcolor="#FFFFFF"><?php echo $array[$1][0]:?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $array[$i][1]:?></td> 
<td height="22" align="center" bgcolor="#FFFFFF"><?php echo $array[$i][2];?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $array[$i][3]:?></td> 
</tr> 
<?php 
} 


?> 


运行 本 示例 ， 循 环 输出 数据 库 中 的 数据 ， 其 运行 结果 如 图 9.23 所 示 。 
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图 9.23 ADODB 操作 数据 库 类 


| SQL 语句 的 类 型 分 为 查询 (select)、 添 加 (insert)、 更 新 (update) 和 删除 (delete) 4 种 。 
| 这 4 个 英文 单词 都 有 一 个 共同 点 ， 都 是 6 个 英文 字母 。 我们 就 是 根据 它们 的 这 个 共同 点 对 SQL 
| 语句 的 类 型 进行 判断 的 , 这 也 就 是 应 用 substr0 函 数 对 SQL 语句 的 前 6 个 字符 串 进行 截取 的 原因 。 


问题 213 ”如何 封装 ADODB 分 页 类 ? 


问题 章 述 


| 通过 ADODB 操作 数据 库 中 的 数据 时 ， 常 常会 需要 对 输出 的 数据 进行 分 页 输出 ， 如 果 在 一 
| 个 程序 中 需要 多 次 应 用 分 页 输出 ， 则 可 以 将 分 页 方法 封装 到 类 中 ， 这 样 就 可 以 在 需要 时 直接 调 
| 用 , 不 必 重 复 编 写 。 那 么 如 何 封装 ADODB 分 页 类 ， 使 之 在 需要 应 用 分 页 类 时 直接 调用 即 可 ? 


| 专家 解答 
| 在 ADODB 分 页 类 SepPage0 中 主要 整合 了 下 面 几 个 ADODB 中 的 函数 。 
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(1) PageExecuteO 函 数 ， 分 页 功能 函数 ， 其 语法 如 下 : 
PageExecute($sql, $nrows, $page); 
参数 说 明 : 
@ $sql: SQL 查询 语句 。 
@ $nrows: 每 页 显示 的 记录 数 。 
目 $page: 保存 当前 页 数 ， 默 认为 1。 
(2) AbsolutePage() 函 数 ， 返 回 当前 页 的 页 数 。 其 语法 如 下 : 
AbsolutePage($page—-1) 
要 与 PageExecute0) 函 数 配 合 使 用 。 
(3) AtFirstPage() 函 数 ， 如 果 当 前 页 是 第 一 页 ， 则 返回 Tme。 其 语法 如 下 : 
AtFirstPage($status=") 
要 与 PageExecute() 函 数 配 合 使 用 ， 判 断 当 前 是 否 是 第 一 页 。 
(4) AtLastPage() 函 数 ， 判 断 当 前 页 是 否 是 最 后 一 页 。 其 语法 如 下 : 
AtLastPage($status="); 
同样 要 与 PageExecute() 函 数 配 合 使 用 ， 判 断 当 前 页 是 否 是 最 后 一 页 ， 如 果 是 则 返回 True。 
应 用 示例 
本 示例 封装 一 个 ADODB 分 页 类 , 并且 运用 ADODB 分 页 类 输出 MySQL 数据库 db_database 
中 tb_user 数据 表 中 的 数据 。 其 具体 步骤 如 下 : 
(1) 创建 found.database.php 文件 ， 定 义 SepPage 分 页 类 及 ShowData() 方 法 ， 读 取 数 据 库 
中 存储 的 数据 ， 并 且 将 查询 结果 返回 到 数组 中 ; 定义 ShowPage() 方 法 ,创建 分 页 超 链 接 ， 完 成 
从 数据 库 中 读 取 数据 的 分 页 输出 。SepPage 分 页 类 的 关键 代码 如 下 : 


class SepPage{ 
Var $rs; 
Var $pagesize; 
Var $nowpage; 
Var $array; 
Var $conn; 
Var $sqlstr; 
function ShowData($sqlstr,$conn.$pagesize,$nowpage){ ”// 定 义 方法 
if(!isset($nowpage) || $nowpage—"") /判断 变量 值 是 否 为 空 
$this->nowpage=1: /定义 每 页 起 始 页 
else 
$this->nowpage=$nowpage: 
S$this->pagesize=$pagesize:; // 定 义 每 页 输出 的 记录 数 
S$this->conn=$conn; // 连 接 数 据 库 返回 的 标识 
Sthis->sqlstr=$sqlstr: /执行 的 查询 语句 
Sthis->rs=$this->conn->PageExecute($this->sqlstr,$this->pagesize,$this->nowpage); 
@$this->array=$this->rs->GetRows(); /1/ 获 取 记 录 数 
这 count($this->aray) 一 0 || $this->rs 一 包 lse) 
Tetum false; 
else 


Tetum $this->array: 
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} 
/省 略 了 ShowPage( 方 法 中 的 内 容 
} 
| (2) 创建 connect.database.php 文件 ， 实 例 化 数据 库 连 接 类 ConnDB、 数 据 库 操作 类 AdminDB 
| 和 分 页 类 SepPage。 其 关键 代码 如 下 : 


(3) 创建 index.php 文件 ， 包 含 connect.database.php 文件 ， 调 用 分 页 类 中 的 ShowData0 方 法 ， 
完成 数据 的 分 页 输出 ， 调 用 分 页 类 中 的 ShowPage0 方 法 输出 分 页 的 超 链接 。 其 关键 代码 如 下 : 


<2php 
| require("found.database.php"); // 包 含 类 文件 
| $connobj=new ConnDB("mysql","localhost","root","111","db_database",false); // 数 据 库 连接 类 实例 化 
| $conn=$connobj->GetConnId(); 
! $admindb=new AdminDB(): /| 数据 库 操作 类 实例 化 
| $seppage=new SepPage(); // 分 页 类 实例 化 
| > 


| <?php 
| include_once 'conn/connect.database.php'; // 调 用 类 中 方法 
| Sarray=$seppage->ShowData("select * from tb_user",$conn,10,$_GET["page"]); 
| // 分 页 读 取 数 据 库 中 的 数据 
| for($i=0;$i<count($array): $i++){ 1/ 循环 输出 数据 库 中 的 数据 
| > 

<tr> 


<td height="20" align="center" bgcolor="#FFFFFF"><?php echo $array[$i][0];?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $array[$i][1]:?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $array[$i][2];?></td> 
</tr> 
<2php 
} 
| ?> 
| <tr> 
| <td height="35” ><?php echo $seppage->ShowPage(" 用户"" 个 "" 参 数 1"" 参 数 
| 2""al");2></td> 
</tr> 


运行 结果 如 图 9.24 所 示 。 


共有 用 户 全 个 每 页 显示 16 个 第 2 页 共 9 页 首页 上 一 页 下 一 页 是 页 


图 9.24 ADODB 分 页 类 的 应 用 


| 在 运用 ADODB 分 页 类 进行 数据 的 分 页 输出 时 , 向 ShowData0 方 法 中 传递 了 4 个 参数 : SQL 
| 查询 语句 、 数 据 库 的 连接 标识 、 每 页 显示 的 记录 数 和 当前 页 码 ， 向 ShowPage0 方 法 中 传递 了 5 
| 个 参数 : 数据 的 类 型 、 数 据 的 单位 、 超 链接 的 参数 一 、 参 数 二 和 超 链接 样式 。 
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问题 214 
问题 215 
问题 216 
问题 217 
问题 218 
问题 219 
问题 220 
问题 221 
问题 222 


Ajax 不 刷新 页 面 提交 表单 数据 

Ajax 无 刷新 验证 用 户 名 是 否 被 占用 

如 何 应 用 Ajax 技术 实现 无 刷新 下 拉 列 表 ? 

如 何 实现 Ajax 无 刷新 分 页 ? 

Ajax 无 刷新 倒计时 

如 何 让 Ajax 中 的 方法 熟悉 兼容 各 种 不 同 的 浏览 器 ? 
jQuery 特效 复 选 框 的 制作 方法 

如 何 使 用 jQuery 技术 制作 级 联 菜单 ? 

如 何 通过 jQuery 生成 网 页 特效 ? 


需 


全 和 
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问题 214 Ajax 不 刷新 页 面 提交 表单 数据 


问题 阐述 


用 户 登 录 功 能 的 原理 是 在 客户 端 通过 表单 提交 用 户 名 和 密码 ， 将 数据 提交 到 服务 器 中 ,在 
服务 器 中 完成 对 提交 用 户 名 和 密码 的 验证 ， 从 而 判断 这 个 用 户 是 否 可 以 登录 。 这 是 该 功能 实现 


| 的 基本 原理 ， 但 是 如 果 通 过 Ajax 技术 来 实现 该 功能 就 不 需要 刷新 页 面 ， 或 者 说 不 需要 重新 加 


载 程序 就 可 以 完成 用 户 名 和 密码 的 验证 ， 从 而 减少 了 刷新 页 面 的 等 待 时 间 。 那么 如 何 实现 Ajax 
| 不 刷新 页 面 提交 表单 数据 呢 ? 


| 专家 解答 


1，Ajax 工作 原理 
应 用 Ajax 技术 实现 用 户 登录 的 功能 ， 其 中 Ajax 的 工作 原理 如 图 10.1 所 示 。 


用 户 发 送 请 求 


请 求 服务 器 
< 


"WW 


一 一 一 一 一 
服务 器 响应 Ajax 响应 


客户 端 浏 览 器 


二 | 


次 
雍 


图 10.1 Ajax 工作 原理 
通过 客户 端 浏览 器 发 送 请 求 〈 登 录用 户 名 和 密码 )， 通 过 Ajax 直接 请 求 服务 器 ， 服 务 器 响 


应 〈 检 测 登 录用 户 名 和 密码 )， 最 终 通过 Ajax 返回 到 客户 端 (输出 验证 结果 )。 


通过 Ajax 技术 实现 用 户 登 录 功 能 ， 其 关键 就 是 XMLHttpRequest 的 应 用 , 它 是 一 个 具有 应 


用 程序 接口 的 JavaScript 对 象 ， 能 够 使 用 超 文本 传输 协议 CHITP) 连接 一 个 服务 器 ， 是 微软 公 
| 司 为 了 满足 开发 者 的 需要 ， 于 1999 年 在 下 5.0 浏览 器 中 率先 推出 的 。 


使 用 XMLHttpRequest 对 象 ，Ajax 可 以 像 桌面 应 用 程序 一 样 只 与 服务 器 进行 数据 层面 的 交 


| 换 ， 而 不 用 每 次 都 刷新 页 面 ， 也 不 用 每 次 都 将 数据 处 理 的 工作 交 给 服务 器 来 做 ， 这 样 妈 减 径 了 
| 服务 器 的 负担 ， 又 加 快 了 响应 速度 ， 缩 短 了 用 户 等 待 的 时 间 。 


2. XMLHttpRequest 对 象 的 常用 方法 


(1) open() 方 法 
open() 方 法 用 于 设置 进行 异步 请 求 目 标的 URL、 请 求 方 法 以 及 其 他 参数 信息 ， 语 法 如 下 : 


open("method","URL"[,asyncFlag[,"userName"[, "password"]]]) 
open() 方 法 的 参数 说 明 如 表 10.1 所 示 。 
表 10.1 open() 方 法 的 参数 说 明 
说 明 
用 于 指定 请 求 的 类 型 ， 一 般 为 get 或 post 
用 于 指定 请 求 地 址 ， 可 以 使 用 绝对 地 址 或 者 相对 地 址 ， 并 且 可 以 传递 查询 字符 串 
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参 数 说 了 明 

asyncFla [ 选 参数 ， 用 于 指定 请 求 方式 ， 同 步 请 求 为 tue， 异 步 请 求 为 false， 默 认 情况 下 为 tue 
UserName 选 参数 ， 用 于 指定 请 求 用 户 名 ， 没 有 时 可 省 略 

password [ 选 参数 ， 用 于 指定 请 求 密码 ， 没 有 时 可 省 略 


(2) send() 方 法 | 
send() 方 法 用 于 向 服务 器 发 送 请 求 。 如 果 请 求 声 明 为 异步 ， 则 该 方法 将 立即 返回 ， 和 否则 将 | 
等 到 接收 到 响应 为 止 。 语 法 格式 如 下 : | 
send(content) 
参数 content 用 于 指定 发 送 的 数据 ， 可 以 是 DOM 对 象 的 实例 、 输 入 流 或 字符 串 。 如 果 没 有 
参数 需要 传递 ， 则 可 以 将 其 设置 为 null。 
(3) setRequestHeader() 方 法 
setRequestHeader() 方 法 为 请 求 的 HITP 头 设置 值 。 语 法 格式 如 下 : 
setRequestHeader("label", "value") 
参数 label 用 于 指定 HTTP 头 ; value 用 于 为 指定 的 HTTP 头 设置 值 。 


注意 : 


setRequestHeader() 方 法 必须 在 调用 open() 方 法 之 后 才能 调用 。 


(4) abort( 方 法 
abort() 方 法 用 于 停止 当前 异步 请 求 。 
(5) getAll]ResponseHeaders() 方 法 
getAllResponseHeaders() 方 法 用 于 以 字符 串 形式 返回 完整 的 HTTP 头 信息 ， 当 存在 参数 时 ， 
表示 以 字符 串 形 式 返 回 由 该 参数 指定 的 HTTP 头 信息 。 
3，XMLHttpRequest 对 象 的 常用 属性 
XMLHttpRequest 对 象 的 常用 属性 如 表 10.2 所 示 。 


表 10.2 XMLHttpRequest 对 象 的 常用 属性 
说 了 明 | 
每 个 状态 改变 时 都 会 触发 这 个 事件 处 理 器 ， 通 常会 调用 一 个 JavaScript 函 数 | 
请 求 的 状态 。 有 以 下 5 个 取 值 : | 
0= 未 初始 化 | 


属 性 


onreadystatechange 


readyState 有 
2= 已 加 载 | 

= 交互 中 | 

4= 完成 | 

TesponseText 服务 器 的 响应 ， 表 示 为 字符 串 | 
responseXML 服务 器 的 响应 ， 表 示 为 XML。 这 个 对 象 可 以 解析 为 一 个 DOM 对 象 | 
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说 明 
返回 服务 器 的 HTTP 状 态 码 ， 如 : 
200=" 成 功 " 
202=" 请 求 被 接受 ， 但 尚未 成 功 " 
400=" 错 误 的 请 求 " 
404=" 文 件 未 找到 " 
500=" 内 部 服务 器 错误 " 
statusText 返回 HTTP 状 态 码 对 应 的 文本 
4. XMLHttpRequest 对 象 与 服务 器 的 交互 
XMLHttpRequest 对 象 与 服务 器 的 交互 分 为 以 下 3 个 步骤 。 
(1) 初始 化 XMLHttpRequest 对 象 。 由 于 XMLHttpRequest 不 是 一 个 W3C 标准 ， 所 以 对 
于 不 同 的 浏览 器 ， 初 始 化 的 方法 也 是 不 同 的 。 
正 浏览 器 把 XMLHttpRequest 实例 化 为 一 个 ActiveX 对 象 。 具 体 方法 为 : 
var http_request = new ActiveXObject("Msxml2.XMLHTTP"); 
或 者 
var http_request = new ActiveXObject("Microsoft. XMLHTTP"); 
上 面 语法 中 的 Msxml2.XMLHTTP 和 MicrosoftXMLHTTP 是 针对 正 浏览 器 的 不 同 版 本 而 
进行 设置 的 ， 是 目前 比较 常用 的 两 种 。 
Mozilla、Safari 等 其 他 浏览 器 把 XMLHttpRequest 实例 化 为 一 个 本 地 JavaScript 对 象 。 具体 
方法 如 下 : 
var http_request = new XMLHttpRequestO; 
初始 化 XMLHttpRequest 对 象 的 代码 如 下 : 
var xmlHttp =false; /定义 XMLHttpRequest 对 象 


status 


try{ // 如 果 浏 览 器 支持 XMLHttpRequest 对 象 ， 则 创建 ActiveXObject 对 象 
xmlHttp = new ActiveXObject("Msxml?2.XMLHTTP"): 

} catch (e) { 
try 


xmlHttp = new ActiveXObject("Microsoft. XMLHTTP"): 
} catch (e2) {} 

} 
if (!IxmlHttp && typeof XMLHttpRequest != "undefined") { 

try{ 

xmlHttp = new XMLHttpRequestO: 

}catch(e3){ xmlHttp = false:} 

} 
(2) 设置 请 求 状态 和 返回 处 理 函 数 ， 语 法 格式 如 下 : 
XmlHttp.onreadystatechange=function name: 
其 中 ，xmlHttp 是 XMLHttpRequest 的 对 象 ，function name 是 用 来 处 理 请 求 状态 和 返回 码 
的 函数 名 〈 即 回调 函数 )。 
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(3) 发 送 HITP 请 求 ， 语 法 格式 如 下 : 
xmlHttp.open(send method,url,flag); 


其 中 ,xmlHttp 是 XMLHttpRequest 的 对 象 ;send_method 是 发 送 方法 ,可 以 是 GET0 或 POSTO | 

其 对 应 表单 使 用 的 方法 ; url 是 页 面 要 调用 的 地 址 ; flag 是 一 个 标记 ,如 果 为 Tue， 则 表示 在 等 | 

待 被 调用 页 面 响应 的 时 间 内 可 以 继续 执行 页 面 代 码 ， 反 之 为 False。 | 

5. 应 用 示 斧 

本 示例 应 用 Ajax 技术 实现 用 户 的 无 刷新 登录 功能 。 其 具体 步 又 如 下 : | 

(1) 创建 conn.php 文件 ， 连 接 MySQL 数据 库 服务 器 ， 选 择 指定 的 数据 库 ， 并 且 设 置 数 | 

据 库 编码 格式 为 UTF-8。 | 

(2) 创建 index.php 文件 ， 完 成 用 户 登 录 页 面 的 设计 。 首 先 ， 典 入 JavaScript 脚本 文件 | 

createxmlhttpjs， 初 始 化 XMLHttpRequest 对 象 ， 嵌 入 checkjs 完成 对 用 户 提交 用 户 名 和 密码 的 

验证 ， 并 通过 open() 方 法 调用 reg_chk.php 文件 完成 异步 请 求 ， 同 时 将 Ajax 的 响应 通过 div 返 
回 到 客户 端 。checkjs 的 关键 代码 如 下 : 


function showsimpleO{ // 创 建 主 控制 函数 
if(register.name.value—""){ /判断 用 户 名 是 否 为 空 
Tegister.name.focus(); 
return false'; 
} 
if(register. pwd1.value—""){ /判断 密码 是 否 为 空 
Tegisterpwdl.focus0O; 
TIetum false; 
} 
if(register.yzm.value—""){ /判断 验证 码 是 否 为 空 
Tegister.yzm.focus(); 
return false; 


} 
if(register.yzm.value!=register.yzm2.value){ /判断 输入 的 验证 码 是 否 正 确 
alert(" 效 验 码 输入 错误 !"); 
Tregister.yzm.focus(); 
Tetum false: 
} 
var username = document.getElementById("name").value: // 获 取 表 单 提交 的 用 户 名 | 
var password = document.getElementById("pwd1").value; /获取 密码 | 
var post_method="user="+username+"&pass="+password; /构造 URL 参数 | 


xmlhttp.open("POST","reg_chk php".true): /调用 指定 的 添加 文件 | 

xmlhttp.setRequestHeader("Content-Type","application/xXx-www-form-urlencoded:;"); | 

/设置 请 求 头 信息 ! 

xmlhttp.onreadystatechange=StatHandler: // 判 断 URL 调用 的 状态 值 并 处 理 | 

xmlhttp.send(post_ method): /将 数据 发 送 给 服务 器 | 

} | 

function StatHandlerO{ // 定 义 处 理 函数 | 

这 xmlhttp TeadyState 一 4 && xmlhttp status 一 200){ /判断 如 果 执 行 成 功 ， 则 输出 下 面 内 容 | 

document.getElementById("webpage").innerHTML=xmlhttp.response Text; | 

/将 服务 器 返回 的 数据 定义 到 DIV 中 | 

} | 

} | 
“313. \ 
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So 
function chkyzm(form){ // 对 验证 码 进行 验证 
if(form.yzm.value—""){ 
yzml.innerHTMIL="<font color=# 本 F0000> 请 输入 效 验 码 ! </font>"; 
}else if(form.yzm.value!=form.yzm?2.value){ 
yzml.innerHTMIL="<font color=# 枉 F0000> 效 验 码 输入 错误 !</font>"; 
jelse{ 
yzml.innerHTML="<font color=green> 输 入 正确 </font>"; 
} 
function yzm(form){ // 生 成 验证 码 
var num1=Math.round(Math.random()*10000000); 
varnum=numl.toStringO.substr(0.4): 
document.write("<img name=codeimg src= yzm.php?num="+Hnum+"> 
form.yzm?2.value=num:; 
h 
function code(form){ // 重 置 验证 码 
var num1=Math.round(Math.random()*10000000):; 
var num=numl.toString0.substr(0.4); 
document.codeimg.stre="yzm.php?num="+num:; 
form.yzm?2.value=num; 


} 
(3) 在 reg_chk.php 文件 中 对 POSTO 方 法 提交 的 用 户 名 和 密码 进行 验证 ， 并 输出 验证 结 


| 果 。 其 代码 如 下 : 


</a>": 


户 填 ? 


<2php 

session_ start(); 

header('Content-type: text/html:charset=utf-8"); 

include("conn.php"); 

$sql = "select * from tb_user where name=".$ POST[user]." and password=".md5($_POST['pass"])."™"; 


// 定 义 SQL 语句 
$mysql_query=mysql_query($sql,$conn):; // 执 行 select 查询 语句 
$result=mysql_num rows($mysql_query); 
if($result<1){ 

echo "登录 失败 ， 用 户 名 或 密码 错误 ， 请 重新 登录 "; 
}else{ 


$_SESSION['member] =$_POST['user']: 
echo $ POST[user]." 登 录 成 功 ""&nbsp:&nbsp:&nbsp:&nbsp:"."<a hre 伍 坟 > 由 此 进入 主页 
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(4) 在 index.php 文件 中 创建 表单 ， 并 提交 用 户 名 和 密码 ， 其 中 ， 通 过 onBlur0 事 件 对 用 
了 的 数据 进行 验证 , 通过 onClick0 事 件 调 用 showsimple0 方 法 完成 用 户 登录 的 验证 。 最 后 ， 


创建 div 标签 ， 输 出 Ajax 响应 的 结果 。index.php 的 关键 代码 如 下 : 


<link rel="stylesheet" href="css/reg.css"/> 
<script language="javascript" src="js/createxmlhttp.js"></script> 
<script language="Javascript" src="js/check.js"></script> 
<table width="547" border="0" align="center" cellpadding="0" cellspacing="0"> 
<form id="register" name="register" method="post" > 
<tr> 
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<td width="108" height="25"><div align="right"> 会 员 名 : </div></td> 
<td height="25" colspan="3">&nbsp; 
<input id="name" name="name" type="text" onMouseOver="this.style.backgroundColor= 
ffffPFP" onMouseOut="this.style.backgroundColor='#e8f4 人 f" />&nbsp;<font color="red">*</font></td> 
</tr> 
<tr> 
<td height="25"><div align="right"> 密 &nbsp; 码 : </div></td> 
<td height="25" colspan="3">&nbsp; 
<input 1d="pwd1" name="pwd1" type="password" onMouseOver="this.style 
.backgroundColor#fffffP” onMouseOut="this.style.backgroundColor='#e8f4 人 ff"/>&nbsp:<font color="red">* 
</font></td> 
</tr> 
<tr> 
<td height="25"><div align="right"> 验 证 码 : </div></td> 
<td width="78" height="25">&nbsp; | 
<input id="yzm" type="text" name="yzm" size="8" onBlur="javascript:chkyzm | 
(register)" onMouseOver="this.style.backgroundColor=#fHP" onMouseOut="this.style.backeroundColor=#e8f4f"/> 
<input name="yzm2" type="hidden" value="" /> 
</td> 
<td width="63" align="center" valign="middle"><script>yzm(register)</script></td> 
<td width="108"><a href="javascript:code(register)">&nbsp;&nbsp; 看 不 清 </a></td> 
<td height="25"><div id="yzm1"><font color="#999999"> 输 入 验证 码 </font></div></td> 
</t> 
<tr> 
<td height="50" colspan="5" align="center" valign="bottom">&nbsp:; 
<img src="images/03-11(2).jpg" width="91" height="31" onclick="showsimpleO;" > 
<input type="image" name="imageField2" src="images/03-11(3).jpg" onclick= | 
"form.reset();return false;" /></td> 
</t> 
</form> 
<tr> 
<td height="50" colspan="5" align="center" valign="bottom"><div id="webpage" /></td> 
</tr> 
</table> 


用 户 无 刷新 登录 的 运行 结果 如 图 10.2 所 示 。 
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图 10.2 用 户 无 刷新 登录 界面 
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问题 215 Ajax 无 刷新 验证 用 户 名 是 否 被 占用 


全 问题 辣 术 


在 一 些 网 站 的 注册 功能 中 ， 为 了 避免 出 现 同名 的 用 户 名 ， 在 注册 时 就 对 用 户 填写 的 用 户 名 
| 直接 进行 验证 ， 如 果 用 户 名 没有 被 占用 ， 则 可 以 使 用 ， 和 否则 ， 将 提示 该 用 户 名 已 被 占用 ， 那 么 
| 用 哪 种 方法 对 用 户 名 验证 最 理想 呢 ? 


专家 解答 


| 用 户 名 验证 功能 的 实现 采用 Ajax 技术 是 最 理想 的 ， 因 为 其 可 以 实现 异步 请 求 ， 无 须 重新 
| 加 载 页 面 即 可 直接 获取 验证 结果 。 
| 通过 Ajax 技术 实现 无 刷新 验证 用 户 名 是 否 被 占用 , 应 用 XMLHttpRequest 对 象 实现 异步 请 
| 求 ， 调 用 指定 的 文件 完成 用 户 名 的 验证 操作 。 有 关 Ajax 技术 的 详细 讲解 ， 读 者 可 以 参考 问题 
| 214 中 的 内 容 ， 这 里 不 再 袭 述 。 
| 应 用 示例 

在 本 示例 中 应 用 Ajax 技术 实现 无 刷新 验证 用 户 名 是 否 被 占用 。 其 实现 过 程 如 下 : 
| (1) 创建 数据 库 连 接 文件 conn.php， 连 接 数据 库 服务 器 (用 户 名 为 root， 密 码 为 111)， 
| 选择 db_database 数据 库 ， 设 置 数据 库 编码 为 UTF-8。 
| (2) 创建 js 文件 夹 ， 编 写 JavaScript 脚本 文件 createxmlhttp.js， 初 始 化 XMLHttpRequest 
| 对象， 编写 checkjs 完成 对 用 户 提交 注册 信息 进行 验证 ， 并 通过 open0 方 法 调用 chkname.php 
| 文件 完成 异步 请 求 , 验证 用 户 填写 的 用 户 名 是 否 被 占用 ,并 将 Ajax 的 响应 通过 div 返回 到 客户 
| 端 ， 同 时 定义 yzm0 和 code0 两 个 方法 ， 完 成 验证 码 的 创建 和 无 刷新 更 换 。 


checkjs 的 关键 代码 如 下 : 
function chkname(form){ 
if(form.name.value—""){ /判断 用 户 名 表单 元 素 值 是 否 为 空 
namel.innerHTML="<font color=#FF0000> 请 输入 用 户 名 ! </font>"; 
// 如 果 为 空 则 输出 该 内 容 
}else{ 
var user = form.name.value: // 否 则 获取 用 户 名 
var Url = "chkname.php?user="+user; /定义 url 
xmlhttp.open("GET",url,true): // 通 过 get0 方 法 执行 
xmlhttp.onreadystatechange = functionO{ // 回 调 函数 
ifxmlhttp readyState — 4){ /判断 返回 值 
var msg = xmlhttp responseText: // 获 取 返 回 值 
imsg — 2){ // 判 断 如 果 返 回 值 为 2 
namel.innerHTMIL"<font color=#EF0000> 用 户 名 被 占用 ! </font>"; 
// 则 说 明 用 户 名 被 占用 
return false: 
jelseifmsg 一 '1){ // 判 断 如 果 返 回 值 为 1 
namel.innerHTMIL="<font color=green> 恭 喜 您 ， 可 以 注册 !</font>"; 
// 则 说 明 用 户 名 可 用 


form.c name.value = "yes"; 
Jelse{ 
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namel.innerHTML="<font color=green>"+msg+"</font>"; 


// 输 出 可 用 用 户 名 
} 
} 
上 
xmlhttp.send(null); // 执 行 发 送 
加 
} 
function chkinput(form){ // 验 证 表单 中 的 元 素 是 否 为 定 ， 是 否 符合 要 求 | 
if(form.name.value==""){ | 
form.name.focus(); | 
return false; | 
这 form.c name.value — "not"){ | 
form.name.select(); | 
Tetum false; | 
} | 
/省 略 了 部 分 代码 | 
) 
function chkpwdl(form){ /无 刷新 验证 注册 信息 ， 并 返回 验证 结果 | 
这 formm.pwdl.value 一 ""){ | 
pwdll.innerHTML="<font color=#FF0000> 密 码 格式 错误 ! </font>"; | 
}else if(form pwdl value length<6){ | 
pwdll.innerHTML="<font color=#FF0000> 注 册 密码 长 度 应 大 于 6 位 ! </font>"; | 
Jelse{ | 
pwd11.innerHTML="<font color=green> 输 入 正确 </font>"; | 
} | 
} | 
/省 略 了 部 分 代码 | 
function yzm(form){ // 创 建 验证 码 | 
var num1=Math.round(Math.random()*10000000); | 
varnum=numl.toString(O.substr(0.4): | 
document.write("<img name=codeimg sre='yzm.php?num="+num+">"); | 
form.yzm2.value=num; | 
} | 
function code(form){ // 验 证 码 的 无 刷新 更 换 | 
var numl=Math round(Math randomO)*10000000):; | 
var num=numl.toStringO.substr(0.4); | 
document.codeimg.ste="yzm.php?num="+num; | 
form yzm2.value=num: | 
} | 


在 checkjs 文件 中 调用 chkname.php 文件 完成 对 用 户 名 的 验证 ， 其 传递 参数 是 用 户 提交 的 
用 户 名 。 调用 yzm.php 文件 完成 验证 码 的 生成 和 无 刷新 更 换 , 其 传递 参数 是 随机 生成 的 验证 码 。 | 
chkname php 的 关键 代码 如 下 : 


aphp 
session start(); | 
include("conn.php"): | 
Sreback = '0'; // 定 义 变 量 | 
$sql = "select * from tb_user where name=".$_ GET[mser]."": /定义 SQL 语句 | 
$mysql_ query=mysql_ query($sql,$conn); /执行 select 查询 语句 | 
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在 yzmphp 文件 


div 标签 输出 验证 结果 ， 
据 库 中 。 其 关键 代码 如 下 : 


FED 
Neu 
De 
$result=mysql_ num rows($mysql query); 
if($result>0){ 
Sreback = "2'; 
Jelse{ 
Sreback ="1'; 


} 
echo $reback; 
?> 


< 
srand((double)microtime()*1000000); 
S$im=imagecreate(60,20); 
$black=imagecolorallocate($im.0,0,0); 
$white=imagecolorallocate($im,255,255,255); 
$egray=imagecolorallocate($im,200,200,200); 
imagefill($im,0,0,$gray); 
for(G$i=0:$i<4:$i+H){ 

$str=mt_rand(1,5):; 

$size=mt_rand(6,9); 

$authnum=substr($_GET[num],$i,1); 


for($i=0:$i<200:$i+H){ 
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// 则 为 变量 赋值 为 2 
// 如 果 查 询 结果 不 为 真 ， 则 为 变量 赋值 为 1 
// 输 出 变量 值 


ph， 以 生成 的 随机 数 为 根本 , 通过 GD2 函数 生成 数字 验证 码 ， 其 代码 如 下 : 


// 生 成 随机 数 

// 创 建 画 布 

/定义 背景 

/定义 背景 

/定义 背景 

/填充 颜 色 

/定义 4 位 随机 数 

/定义 随机 字符 所 在 位 置 的 的 立 坐 标 
/定义 随机 字符 的 字体 

/获取 超 链接 中 传递 的 验证 码 


imagestring($inm.$size,(3+$i*15),$str$authnunmimagecolorallocate($inrand(0.130),rand(0,130)rand(0,130)7); 
} 


// 水 平 输出 字符 串 
// 执 行 for 循环 ， 为 验证 码 添加 模糊 背景 


S$randcolor=imagecolorallocate($im,rand(0,255),rand(0,255),rand(0,255)); // 创 建 背 景 


imagesetpixel($im,rand()%70,randO)%30,$randcolor):; 


} 

imagepng($im); 
imagedestroy($inm); 
?> 


/绘制 单一 元 素 


/生成 png 图 像 
/销毁 图 像 


(3) 创建 index.php 文件 ， 设 计 用 户 注册 页 面 。 首 先 通 过 script 标签 调用 js 脚本 文件 ， 然 


后 创建 form 表单 ， 并 在 该 表单 中 通过 JavaScript 脚本 事件 完成 对 用 户 注册 信息 的 验证 ， 通 过 
同时 将 注册 信息 提交 到 reg_chk.php 文件 中 ， 将 用 户 注册 信息 添加 到 数 


<script language="javascript" src="js/createxmlhttp.js"></script> 
<script language="javascript" src="js/check.js"></script> 
<body onLoad='""javascript:registername.focus0"> 
<table width="547" border="0" align="center" cellpadding="0" cellspacing="0"> 
<form id="register"” name="register" action="reg_chk.php" method="post" onSubmit="return 


chkinput(this)"> 


<tr> 


<td width="108" height="25"><div align="right"> 用 户 名 : </div></td> 


<td height="25" colspan="3">&nbsp; 


<input id="name" name="name" type="text" onBlur="javascript:chkname(register)" 


‘onMouseOver="this.style.backgroundColor='#ffffr" on MouseOut="this.style.backgroundColor='#e8f4 人 ff" /> 


<input id="c name" name="c anme" type="hidden" value="not" >&nbsp;<font 


color="red">*</font></td> 


<td height="25"><div id="name1"><font color="#999999"> 请 输入 用 户 名 </font></div></td> 
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</tr> 
<tr> 
<td height="25"><div align="right"> 验 证 码 : </div></td> 
<td width="78" height="25">&nbsp; 
<input 1d="yzm" type="text" name=' " size="8" onBlur="javascript:chkyzm(register)" 
‘onMouseOver="this.style.backeroundColor='#fffff" on MouseOut="this.style.backeroundColor='#e8f4 人 f"/> 
<input name="yzm2" type="hidden" value="" /></td> 
<td width="63" align="center" valign="middle"><script>yzm(register)</script></td> 
<td width="108"><a href="javascript:code(register)">&nbsp;&nbsp; 看 不 清 </a></td> 
<td height="25"><div id="yzm1"><font color="#999999"> 输 入 验证 码 </font></div></td> 
</tr> 
<tr> 
<td height="50" colspan="5" align="center" valign="bottom">&nbsp; 
<input type="image" name="imageField" src="images/03-11(2).jpg" />&nbsp;&nbsp; 
<input type="image" name="imageField2" src="images/03-11(3).jpe" onclick="form.reset|; 
return false;" /></td> 
</tr> 
</form> 
</table> 


(4) 创建 reg_chk.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 并 将 用 户 注册 信息 添加 到 指定 的 数 
据 表 中 。 其 运行 结果 如 图 10.3 所 示 。 
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图 10.3 Ajax 无 刷新 验证 用 户 名 是 否 被 占用 
问题 216 ”如 何 应 用 Ajax 技术 实现 无 刷新 下 拉 列 表 ? 


问题 前 述 
Ajax 技术 可 以 应 用 到 用 户 登 录 和 用 户 注册 的 实现 中 , 那么 该 技术 能 否 在 下 拉 列表 中 使 用 呢 ? 
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Ajax 技术 是 可 以 在 下 拉 列 表 中 使 用 的 ， 如 在 下 拉 列 表 A 中 选择 一 个 指定 的 值 ， 当 鼠标 光 
标 失去 焦点 并 发 生变 化 时 ， 会 在 下 拉 列 表 B 中 输出 一 个 与 下 拉 列 表 A 对 应 的 值 。 其 原理 是 根 
据 下 拉 列 表 A 中 值 的 变化 ， 下 拉 列 表 B 中 也 同时 生成 一 个 对 应 的 值 。 在 这 个 过 程 中 ， 下 拉 列 
表 A 中 的 值 是 从 数据 库 中 读 取 的 ， 并 且 通 过 onChange0 事 件 调 用 JavaScript 脚本 函数 ， 应 用 
Ajax 将 下 拉 列 表 的 值 传递 到 指定 的 文件 中 ， 根 据 传递 的 值 在 指定 的 文件 中 生成 下 拉 列 表 B 的 


| 值 ， 最 后 通过 Ajax 将 下 拉 列 表 B 的 值 返 回 到 客户 端 。 


应 用 示例 
在 本 示例 中 ， 当 用 户 选择 左 侧 的 “所 属 大 类 ”时 ， 随 之 右 侧 “ 所 属 小 类 ”将 出 现 大 类 中 对 


| 应 的 小 类 。 其 运行 结果 如 图 10.4 所 示 。 
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图 10.4 Ajax 实现 无 刷新 的 下 拉 列 表 


具体 实现 的 过 程 如 下 : 
(1) 创建 index.php 文件 ， 完 成 图 书信 息 添 加 页 面 的 设计 。 首 先 通过 script 标签 调用 js 脚 


本 文件 ， 用 于 完成 Ajax 的 异步 请 求 和 对 表单 元 素 值 的 验证 。 然 后 创建 form 表单 ， 设 置 下 拉 列 
| 表 完 成 图 书 类 别 输出 ， 并 通过 onChange 事件 调用 js 文件 中 的 函数 ， 实 现 Ajax 异步 请 求 ， 完 成 


下 拉 列 表 的 关联 。 最 后 通过 onClick 事件 调用 js 文件 中 的 函数 ， 对 表单 中 的 元 素 值 进 行 判断 ， 
完成 数据 提交 。 其 关键 代码 如 下 : 
<script language="javascript" sre="js/reg.js"></script> 
<table width="575" border="0" align="center" cellpadding="-2" cellspacing="-2" bordercolordatk= 
"#FFFFFF"> 
<form action="" method="post" name="form1"> 
<tr> 
<td height="27" align="right">&nbsp: 所 属 大 类 : </td> 
<td width="31%" height="27">&nbsp; 
<select name="type" class="textarea" id="type" onChange="F_super(this.value)"> 


<?php 
S$link = mysql_connect ( "localhost", "root", "root" ); /连接 数据 库 服务 器 
Imysql select db ( "db _database", $link ): /连接 数据 库 
//Ajax 中 先 用 encodeURIComponent 对 要 提交 的 中 文 进行 编码 
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$GB2312string = iconv ('UTF-8', 'gb2312//IGNORE,', $SRequestAjaxString ); 
mysql query ("set names gb2312" ); /设置 数据 库 编码 格式 
$sql = mysql query ( "select * from tb_type group by type" ); /执行 查询 操作 
$result = mysql fetch_object ( $sql ): // 获 取 结 果 集 中 的 一 行 记录 
do { /循环 输出 查询 结果 ， 将 结果 作为 下 拉 列 表 的 值 代 站 
header ( 'Content-type: text/html:charset=GB2312' ); /指定 发 送 数据 的 编码 格式 为 GB2312 ; ”二 
?> 
<option value="<?php echo $result->type: ?>" selected><?php echo $result->type: ?> </option> 
<2php | 
} while ( $result = mysql_fetch_object ( $sql ) ); | 
?> ! 
</select> 
</td> 
<td width="13%" height="27" align="right">&nbsp; 所 属 小 类 : </td> 
<td width="42%" height="27" id="subType">&nbsp;</td> 
</tr> 
<tr> 
<td height="28" colspan="4" align="center"> 
<input name="Button"” type="button"” class="btn grey” value=" 保 存 " onClick= 


"mycheckO:"> 
<input name="Submit2" type="reset" class="btn_grey" value=" 重 置 "> 
</td> 
</tr> 
</form> 
</table> 
<script language="javascript">F_super(form!] .type.value):</script> 


(2) 创建 js 文件 夹 ， 编 写 regjs 脚本 文件 ， 定 义 createRequest0) 函 数 ， 实 现 Ajax 的 异步 
请 求 ， 定义 alertContents() 函 数 ， 作 为 回调 函数 ， 返 回 服务 器 的 响应 ;定义 F_super0 函 数 ， 调 
用 createRequest() 函 数 , 将 selSubType.php 文件 和 下 拉 列 表 的 值 作为 参数 ; 定义 mycheck0 函 数 ， 
对 表单 中 的 元 素 值 进行 判断 ， 并 执行 提交 操作 。 其 关键 代码 如 下 : 


Var http_request = false; 


function createRequest(url) { /初始 化 对 象 并 发 出 XMLHttpRequest 请 求 
http_request = false; 
if (window.XMLHttpRequest) { /Mozilla 或 其 他 除 正 以 外 的 浏览 器 


http_request = new XMLHttpRequestO; | 
if (http_request.overrideMimeType) { | 
http_request.overrideMimeType("text/xml"); 


} | 

} else if (window.ActiveXObject) { // 正 浏览 器 | 
ty { | 
http_request = new ActiveXObject("Msxml2.XMLHTTP"): | 

} catch (e) { | 
try{ | 


http_request = new ActiveXObject("Microsoft XMLHTTP"); | 
} catch (e) © 


) 
if (Ihttp_request) { | 
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| alert(" 不 能 创建 XMLHTTP 实例 1"); 
| Tetum false; 
| } 
| http request.onreadystatechange = alertContents; /指定 响应 方法 
http requestopen("GET", url, true): /发 出 HITP 请 求 
http_request.send(null); // 执 行 发送 
上 
function alertContentsO { // 处 理 服务 器 返回 的 信息 
if (http_request.readyState — 4) { 
if (http_request.status 一 200) { 
subType.innerHTMIL=http_request.responseText; // 将 服务 器 返回 值 赋 给 指定 ID 
}else{ 
alert( 您 请 求 的 页 面 发 现 错误 ); 
} 
} 


function F_super(vaD){ 
createRequest("selSubType.php?type="+val+'&nocache='+new Date().getTime()); 


// 实 现 级 联 下 拉 列 表 
} 
function mycheckO{ 
if (form!l.introduce.value—""){ 
alert(" 请 输入 商品 简介 ! ");forml.introduce.focus(:return; 
forml.submitO; 
} 


| (3) 创建 selSubType.php 文件 ， 根 据 Ajax 异步 请 求 中 提交 的 下 拉 列 表 选 项 的 值 ， 从 数据 
| 库 中 读 取 出 对 应 的 数据 ， 并 且 将 读 取 的 数据 作为 下 拉 列表 的 值 ， 重 新 生成 一 个 下 拉 列 表 。 其 关 


| 键 代 码 如 下 
| <2php 
| S$link=mysql_connect("localhost","root","root"); // 连 接 数 据 库 服务 器 
| mysql_select_db("db_database",$link): // 连 接 数 据 库 
| /Ajax 中 先 用 encodeURIComponent 对 要 提交 的 中 文 进 行 编码 
| $GB2312string=iconv( 'UTF-8', 'gb2312//IGNORE,',$RequestAjaxString):; 
| mysql_query("set names gb2312"); // 设 置 编码 格式 
| $type=$_GET[typel; /获取 Ajax 中 传递 的 值 
| $sql=mysql_query("select * from tb_type where type='$type'"); /执行 查询 操作 
| $result=mysql fetch_array($sql): // 获 取 查询 结果 
| header('Content-type: text/html;charset=GB2312"); /指定 发 送 数据 的 编码 格式 为 GB2312 
| ?> 
| <select name="typeID" class="textarea" 1d="typeID"> 
<apmp 
| dof 
| ?> 
| <option value="<2php echo $result[subtype]:2>"><2php echo $result[subtype]:2></option> 
| <2php 
}while($result=mysql_ fetch_array($sqD): 
?> 


</select> 
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问题 217 ”如何 实 现 Ajax 无 刷新 分 页 ? 


问题 阐述 | 鲜 


有 时 人 们 在 线 看 视频 时 ， 会 看 到 下 面 有 很 多 评论 ， 当 查看 评论 时 ， 在 执行 翻 页 的 操作 后 ， 
视频 文件 又 被 重新 打开 ， 由 于 网 络 原因 ， 当 再 次 打开 视频 时 发 现 其 播放 速度 会 变 得 很 慢 ， 那 么 | 
如 何 才 能 实现 在 进行 翻 页 操作 时 ， 不 影响 视频 的 播放 呢 ? 最 理想 的 解决 方案 就 是 应 用 Ajax 技 | 
术 实 现 无 刷新 分 页 。 | 
专家 解答 
所 谓 无 刷新 分 页 ， 就 是 在 进行 翻 页 的 过 程 中 不 需要 重新 加 载 页 面 ， 而 是 在 当前 页 面 中 完成 | 
翻 页 的 操作 。 | 
通过 Ajax 技术 实现 无 刷新 分 页 功能 的 关键 是 XMLHttpRequest 的 应 用 , 它 可 以 像 桌面 应 用 | 
程序 一 样 只 与 服务 器 进行 数据 层面 的 交互 ， 而 不 用 每 次 都 刷新 页 面 ， 也 不 用 每 次 都 将 数据 处 理 | 
的 工作 交 给 服务 器 来 做 ， 这 样 既 减轻 了 服务 器 的 负担 ， 又 加 快 了 响应 速度 ， 缩 短 了 用 户 等 待 的 | 
时 间 。 在 实现 Ajax 无 刷新 分 页 时 将 应 用 到 如 下 所 示 的 Ajax 技术 。 | 
(1) open0 方 法 
open() 方 法 的 具体 内 容 请 参考 问题 214。 
(2) send() 方 法 
send() 方 法 的 具体 内 容 请 参考 问题 214。 
(3) XMLHttpRequest 对 象 的 常用 属性 
XMLHttpRequest 对 象 的 常用 属性 如 表 10.2 所 示 


应 用 示例 


本 示例 在 播放 .flv 文件 的 同时 完成 数据 的 无 刷新 分 页 输出 ， 其 运行 结果 如 图 10.5 所 示 。 


一 


二 i 执 了 之 于 与 了 佬 龙 | 
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图 10.5 Ajax 无 刷新 分 页 


“323® 


~ 人 ip 交 须 知道 的 300 个 问题 
| 在 本 示例 中 实现 的 是 超 长 文本 中 数据 的 分 页 读 取 ， 为 了 体现 Ajax 无 刷新 的 特点 ， 在 页 面 
| 中 还 插入 了 一 个 .hv 格式 的 视频 。 其 具体 实现 步骤 如 下 : 
| (1) 创建 index.php 页 面 。 包括 3 个 方面 的 内 容 : 一 是 嵌入 .fk 格式 的 视频 文件 ， 二 是 完 
”成 超 长 文本 的 分 页 输出 ， 三 是 通过 script 标签 嵌入 js 脚本 文件 ， 定 义 div 标签 ， 为 无 刷新 分 页 
一. 
第 一 方面 ， 嵌 入 .fk 格式 的 视频 文件 ， 其 代码 如 下 : 
<div style="width:520px:; height=185 margin:0px; border:solid Spx #fcedda:"> 
<div id="flash5" class="right">Flvplayer 播放 器 样式 1</div> 
</div> 
<script type="text/javascript" src="js/swfobject.js" ></script><!-- 载 入 js 脚本 文件 --> 
<script type="text/javascript"> 
Var s5 = new SWFObject("FlvPlayerV2009.swf","mediaplayer","520","240","8"); 
s5.addParam("allowfullscreen","true"); 


s5.addVariable("width","520"); // 设 置 宽度 
s5.addVariable("height","240"); // 设 置 高 度 
s5.addVariable("image","images/flash3.jpg"): /设置 背景 
s5.addVariable("file","player.flv"); /指定 播放 文件 
s5.addVariable("backcolor"."Oxff8c00"): // 设 置 背 景 颜 色 

s5.addVariable("frontcolor"."OxE2FOFE"): // 设 置 字 体 颜 色 
s5.write("flash5"):; // 将 内 容 写 入 flash5 

</script> 


第 二 方面 ， 超 长 文本 的 分 页 输出 。 该 技术 和 其 他 分 页 技术 相同 ， 但 是 有 一 点 必须 注意 : 在 
无 刷新 分 页 的 超 链接 中 ，href 的 值 不 是 一 个 具体 的 地 址 ， 而 是 “#”。 无 刷新 分 页 中 通过 onClick 
事件 调用 自 定义 的 方法 进行 异步 请 求 , 而 调用 另外 一 个 文件 执行 页 面 的 跳 转 。 其 关键 代码 如 下 : 
<a href="#" onClick='return no_refurbish_pagination("index_ok.php?page=1")> 首 页 </a>&nbsp; 
<a href="#" onClick='retum no _refinbish pagination("index_ok.php?page=<?php echo $ GET[page]-1:?>> 
上 一 页 </a> 
<a href="#" onClick=retum no Tefurbish pagination("index_ okphp?page=<2php echo $ GET[page]+1:?>)> 
下 一 页 </a> 
<a href="#" onClick='retum no_refurbish pagination("index_okphp?page=<2php echo $page_count:?>")> 
尾 页 </a> 
这 里 调用 的 是 no_refurbish _ pagination() 方 法 执行 异步 请 求 ， 请 求 的 文件 是 index_ok.php， 
| 通过 该 文件 执行 分 页 跳 转 。 
| 第 三 方面 ， 为 Ajax 无 刷新 做 准备 ， 嵌 入 js 脚本 文件 discuss jsjs， 定 义 div 标签 ， 设 置 标 
| 签 的 为 synopsis，div 标签 中 的 内 容 是 在 分 页 中 重新 载 入 的 内 容 。 


| 技巧: 
| 在 <div id="synopsis">...</div> 中 存储 的 必须 是 一 个 完整 的 table 表格 ， 否 则 是 无 法 重新 载 
| 入 的 。 在 这 个 div 中 存储 的 内 容 就 是 异步 请 求 文件 index_ok php 中 的 内 容 。 

| (2) 创建 index_okphp 文件 ， 定 义 异步 请 求 文件 的 内 容 ， 也 就 是 重新 保存 index php 中 
| <div id="synopsis"> 包 含 的 内 容 。 这 个 index_ok php 中 存储 的 是 整个 超 长 文本 的 分 页 读 取 方法 。 


技巧 


在 异步 请 求 文件 index_ok.php 中 需要 设置 页 面 的 编码 格式 ， 否 则 在 重新 载 入 时 可 能 会 出 现 
乱码 的 问题 。 
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(3) 创建 discuss_jsjs 脚本 文件 ,定义 no_refurbish_pagination() 方 法 ,执行 异步 请 求 操作 。 
其 代码 如 下 : | 


xmlHttp = new ActiveXObject("Msxml2.XMLHTTP"):; 
} catch (e) { 
try{ 


xmlHttp = new ActiveXObject("Microsoft. XMLHTTP"): 
} catch (e2) 他 
} 
if (!xmlHttp && typeof XMLHttpRequest != "undefined") { 
try{ 
xmlHttp = new XMLHttpRequestO; 
}catch(e3){ xmlHttp = false;} 


} 
/使 用 XMLHttpRequest 对 象 创建 异步 HITP 请 求 
function no_refurbish pagination(url){ // 创 建 自 定义 函数 ， 获 取 传 递 的 参数 
xmlHttp.open('get',url,true); // 根 据 传递 的 参数 ， 通 过 get0 方 法 执行 异步 请 求 
xmlHttp.onreadystatechange = function( { 
if(xmlHttp.readystate 一 4 && xmlHttp.status — 200){ /返回 结果 
document.getElementById("synopsis").innerHTML = xmlHttp.response Text; 
) 
} 
xmlHttp.send(null); 
b 
专家 点 评 


以 上 就 是 Ajax 无 刷新 分 页 的 步 又， 分 析 其 原理 ， 就 是 在 原 有 分 页 输出 数据 的 基础 上 又 定 
义 了 一 个 相同 的 内 容 ， 而 该 内 容 存储 于 B 文件 中 。 所 谓 的 无 刷新 就 是 利用 了 Ajax 可 以 直接 向 | 
服务 器 发 送 请 求 的 特点 直接 请 求 B 文件 ， 并 将 返回 数据 在 原 有 文件 中 输出 。 | 

在 实现 Ajax 无 刷新 分 页 时 ， 还 需要 注意 以 下 几 点 。 

1. 超 链接 href 值 的 设置 | 

在 通过 Ajax 实现 无 刷新 分 页 的 过 程 中 ,在 创建 分 页 超 链接 时 ，href 的 值 设 置 为 “#”， 通 过 | 
onClick 事件 调用 no_refurbish pagination() 方 法 完成 分 页 操作 。 | 

这 里 必须 要 注意 href 的 值 ， 如 果 此 时 将 href 的 值 设置 为 空 (“”)， 那 么 就 不 能 实现 页 面 跳 | 
转 ， 因 为 其 链接 的 是 当前 页 。 | 

2. no refurbish pagination() 方 法 中 的 参数 | 

onClick 事件 调用 no_refurbish _ pagination() 方 法 完成 分 页 操作 ， 在 no_refurbish _ pagination0) | 
方法 中 传递 了 一 个 参数 ， 即 异步 请 求 的 文件 和 相应 的 参数 值 。 在 no_refurbish_pagination0 方 法 | 
中 定义 该 参数 时 ， 如 果 该 参数 中 存在 PHP 脚本 ， 那 么 必须 要 使 用 双 引 号 或 单 引号 进行 定义 ， | 
其 正确 格式 如 下 : | 

a href="#" onClick tetum no refinbish pagination("index okKphp7page-<7phpechog GET[page 1:2>")> | 

下 一 页 </a> | 

如 果 onClick 事件 使 用 的 是 单 引号 ， 那 么 在 定义 no_refurbish pagination() 方 法 的 参数 时 就 
6s Le 


<a href="#" onClick="return no _refurbish pagination('index_ok.php?page=<?php echo $page_count?> ) "> 
尾 页 </a> 


如 果 onClick 事件 使 用 的 是 双 引 号 ， 那 么 在 定义 no_refurbish pagination() 方 法 的 参数 时 就 
要 使 用 单 引号 。 

如 果 不 使 用 双 引 号 或 单 引号 ， 那 么 将 输出 如 图 10.6 所 示 的 错误 信息 。 

如 果 onClick 事件 使 用 单 引号 ， 而 在 定义 no_refurbish _ pagination() 方 法 的 参数 时 也 使 用 单 


| 引号 ， 或 者 onClick 事件 使 用 双 引 号 ， 而 在 定义 no_refurbish pagination() 方 法 的 参数 时 也 使 用 


| 双 引 号 ， 那 么 将 输出 如 图 10.7 所 示 的 错误 信息 。 


Internet Pxplorer 
请 P9 的 间 题 可 能 亿 其 无 法 正常 呈 志 下 功 认 不 正 芝 ， 以 后 ， 邓 击 显示 在 状 大 


plorer x 
该 同 页 上 上 的 问题 可 欧 舍 式 无 法 正常 显示 或 功能 不 正 管 。 以 后 ,双击 显示 在 状态 


企 栏 中 的 警 洁 图标， 就 可 以 显示 上 述 信息 。 栏 中 的 警 半 医 标 ， 就 可 以 显示 上 述 信息 。 
已 网 页 包 合 请 训 时 妨 经 显示 语 信 息 从)。 厢 同 页 包 合 情 识 肝 她 终 星 示 访 信 息 从。 
[mc TE [snowrau< 


行 ， 3 
字符 51 
请问 缺少 
代码 0 
Es Riegs//127.0.0.1 Maateneaf /te.3 
IE 下 二 TO 
图 10.6 错误 提示 信息 (一 ) 图 10.7 错误 提示 信息 (二) 


问题 218 Ajax 无 刷新 倒计时 


| 问题 请 述 


在 网 页 中 实现 倒计时 非常 普遍 ， 其 原理 也 很 简单 ， 就 是 用 指定 日 期 的 时 间 戳 减 去 当前 日 期 


| 的 时 间 戳 ， 得 到 的 就 是 距离 目标 日 期 的 期 限 。 但 是 当 倒计时 的 时 间 以 分 秒 进行 计算 时 ， 这 个 问 
| 题 似 乎 就 变 得 有 点 “意思 ”了 ， 因 为 毕竟 不 能 通过 手动 来 不 停 地 刷新 网 页 ， 达 到 刷新 倒计时 时 
| 间 的 目的 , 那么 就 必须 实现 倒计时 时 间 的 无 刷新 输出 , 这 就 是 这 里 要 解决 的 问题 一 一 Ajax 无 刷 
| 新 倒计时 。 


| 专家 解答 


既然 要 通过 Ajax 实现 无 刷新 倒计时 ,那么 首先 必须 初始 化 XMLHttpRequest 对 象 ， 并 且 保 


| 证 其 对 不 同 浏览 器 的 支持 。 初 始 化 的 方法 如 下 : 


var xmlHttp = false; 
ty { 
xmlHttp = new ActiveXObject("Msxml2.XMLHTTP"):; 
}catch (e) { 
ty{ 
xmlHttp = new ActiveXObject("Microsoft. XMLHTTP"); 
} catch (e2) 他 
} 


if (IxmlHttp && typeof XMLHttpRequest != "undefined") { 
。326。 
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try 
{ 
xmlHttp = new XMLHttpRequestO; 
}catch(e3){ xmlHttp = false;} 
} | 
然后 调用 XMLHttpRequest 对 象 中 的 open0 方 法 , 执行 异步 请 求 。 open0 方 法 用 于 设置 进行 
异步 请 求 目 标的 URL、 请 求 方法 以 及 其 他 参数 信息 。 具 体 语 法 如 下 : 
open("method","URL"[,asyncFlag[,"userName"[, "password"]]]) | 
在 上 面 的 语法 中 ，method 用 于 指定 请 求 的 类 型 ， 一 般 为 get 或 post; URL 用 于 指定 请 求 的 | 
地 址 ， 可 以 使 用 绝对 地 址 或 相对 地 址 ， 并 且 可 以 传递 查询 字符 串 ，asyncFlag 为 可 选 参数 ,用 于 | 
指定 请 求 方式 ， 同 步 请 求 为 true， 异步 请 求 为 false， 默 认 情况 下 为 true; userName 为 可 选 参 | 
数 ， 用 于 指定 请 求 用 户 名 ， 没 有 时 可 省 略 ，password 为 可 选 参数 ， 用 于 指定 请 求 密码 ， 没 有 时 | 
可 省 略 。 | 
接着 通过 XMLHttpRequest 对 象 中 的 onreadystatechange 属性 调用 JavaScript 脚本 函数 ， | 
通过 responseText 属性 获取 服务 器 响应 返回 的 字符 串 。XMLHttpRequest 对 象 的 常用 属性 如 | 
表 102 所 示 。 | 
最 后 通过 XMLHttpRequest 对 象 中 的 send() 方 法 向 服务 器 发 送 请 求 。 如 果 请 求 声明 为 异步 ， 
则 该 方法 将 立即 返回 ， 和 否则 将 等 到 接收 到 响应 为 止 。 其 语法 如 下 : | 
send(content) 


在 上 面 的 语法 中 ，content 用 于 指定 发 送 的 数据 ， 可 以 是 DOM 对 象 的 实例 、 输 入 流 或 字符 | 

串 。 如 果 没有 参数 需要 传递 ， 则 可 以 设置 为 null。 | 
这 是 无 刷新 倒计时 中 Ajax 部 分 应 用 到 的 技术 、 方 法 。 另 外 ， 还 有 与 PHP 结合 的 内 容 ， 即 | 
通过 open() 方 法 异步 请 求 的 文件 ， 在 请 求 的 文件 中 ， 计 算出 倒计时 的 用 时 及 剩余 时 间 ， 最 终 通 | 
过 responseText 属性 获取 返回 的 时 间 字 符 串 ， 并 且 将 其 赋 给 指定 的 div 标签 ， 在 页 面 中 输出 。 | 
在 计算 倒计时 的 用 时 和 剩余 时 间 时 应 用 mktime0 函 数 获取 系统 的 当前 时 间 戳 、 倒 计时 开始 | 

的 时 间 戳 以 及 倒计时 结束 的 时 间 戳 ， 只 需 对 这 3 个 值 做 减法 运算 ， 即 可 获取 倒计时 的 用 时 和 剩 | 
余 时 间 。 其 关键 代码 如 下 : | 


<?php 


session start(); // 初 始 化 SESSION 变量 
header('Content-Type: text/xml'): // 定 义 页 面 文件 格式 
$dates=$_SESSION['dates']; // 获 取 计 时 开始 时 间 
$dates2=mktime(); // 获 取 当 前 时 间 
$dates3=$dates2-$dates: /计算 倒计时 的 用 时 
echo date("i:s",$dates3); /格式 化 输出 用 时 
?> 

应 用 示例 


应 用 Ajax 实现 无 刷新 倒计时 ， 设 计 倒计时 的 时 长 为 20 分 钟 ， 当 倒计时 结 来 后 给 出 提 示 信 | 
息 。 首先 , 通过 mktime() 函 数 获取 一 个 倒计时 开始 的 时 间 截 , 将 这 个 值 提交 到 倒计时 输出 页 面 ， 
存储 到 SESSION 变量 中 ,在 输出 页 面 中 定义 两 个 JavaScript 脚 本 函数 ShowTime() 和 sparetime()， 
通过 异步 请 求 调用 showtime php 和 sparetime php 这 两 个 文件 , 获取 到 倒计时 的 用 时 和 剩余 时 间 ， | 
最 后 将 获取 的 时 间 在 div 标签 中 输出 。ShowTime0 方 法 的 代码 如 下 : 


Ne pg L_ 


A 人 Pp 六 需 知道 的 300 个 问题 


<script type="text/javascript"> 
timer = window.setInterval("ShowTime()",1000); 
function ShowTimeO{ 
xmlHttp.open("post","showtime.php", true); 
xmlHttp.onreadystatechange = fonction0{ 
if(xmlHttp.readyState — 4){ 
tet = xmlHttp.responseText:; 
document.getElementById("show_time").innerHTML = tet:; 


} 
xmlHttp.send(null); 
1 
</script> 


<div class="STYLE1" id="show_time"></div> 


showtime.php 文件 的 代码 如 上 面 
运行 结果 如 图 10.8 所 示 。 


专家 点 评 


无 刷新 倒计时 运用 Ajax 在 不 需要 
重新 载 入 整个 页 面 的 情况 下 ，XML- 图 10.8 Ajax 无 刷新 倒计时 
HttpRequest 对 象 就 可 以 向 服务 器 发 送 请 求 并 得 到 服务 器 响应 的 特性 , 实现 倒计时 时 间 在 页 面 中 
无 刷新 地 更 新 输出 。 

在 应 用 Ajax 时 ， 由 于 Ajax 不 支持 多 种 字符 集 ， 它 默认 的 字符 集 是 UTF-8， 所 以 在 应 用 
Ajax 技术 的 程序 中 应 及 时 进行 编码 转换 ,避免 在 输出 中 文字 符 串 时 出 现 乱码 。 最 简单 有 效 的 方 
法 就 是 程序 统一 使 用 UTF-8 编码 ， 这 时 就 不 会 涉及 编码 转换 或 者 乱码 的 问题 了 。 


问题 219 ”如 何 让 Ajax 中 的 方法 熟悉 兼容 
各 种 不 同 的 浏览 器 ? 


问题 曾 述 

很 多 时 候 用 户 在 正 浏览 器 中 使 用 Ajax 实现 的 功能 是 正常 的 ， 但 是 当 在 Firefox (火狐 ) 等 
其 他 浏览 器 中 运行 相同 的 程序 时 ， 却 得 不 到 同样 的 效果 ， 甚 至 有 些 功 能 不 能 正常 运行 。 这 就 是 
由 于 Ajax 在 不 同 的 浏览 器 中 使 用 的 方法 不 兼容 导致 的 。 
专家 解答 

解决 Ajax 在 不 同 浏览 器 中 不 兼容 的 问题 ， 其 方法 就 是 自 定 义 一 个 函数 ， 判 断 当 使 用 不 同 
的 浏览 器 时 ， 返 回 不 同 的 内 容 。 其 定义 的 方法 如 下 : 

function getOsO{ 
var OsObject = ""; 


if(navigator.userAgent.indexOf("MSIE")>0) { 
Tetum "MSIE": /IE 浏览 器 


“328's 
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} 
if(isFirefox=navigator.userAgent.indexOf("Firefox")>0){ 
Tetum "Firefox"; //Firefox 浏览 器 


| 
if(isSafari=navigator.userAgent.indexOf("Safari")>0) { 
return "Safari"; //Safari 浏览 器 
} 
if(isCamino=navigator.userAgent.indexOf("Camino")>0){ 
Ietum "Camino"; //Camino 浏览 器 


) 
if(isMozilla=navigator.userAgent.indexOf(“"Gecko/")>0){ 
return "Gecko"; //Gecko 浏览 器 
) 
} 
然后 ， 在 Ajax 中 调用 自 定义 函数 ， 通 过 自 定义 函数 根据 不 同 的 浏览 器 类 型 执行 不 同 格式 | 
的 方法 。 其 关键 代码 如 下 : | 
Var btype=getOs(); 
xmlHttp.onreadystatechange = (btype!="Firefox")?(xmlHandle):(xmlHandle()); 
xmlHttp.open ("GET",CartUrl,false); 
xmlHttp.send(null); 


如 此 Ajax 实现 的 功能 在 下 6、IE7 或 Firefox 浏览 器 下 就 都 可 以 正常 运行 了 。 
问题 220 jQuery 特效 复 选 框 的 制作 方法 


问题 曾 述 | 
表单 中 复 选 框 的 多 选 、 反 选 是 一 个 很 常用 的 方法 ， 经 常 出 现在 数据 的 后 台 管理 中 ， 实 现 多 | 
数据 的 批量 操作 。 其 最 常用 的 实现 方法 就 是 通过 JavaScript 脚本 来 完成 ， 编 写 相当 长 的 一 段 | 
JavaScript 代码 来 实现 这 个 功能 ， 为 了 让 这 个 功能 实现 得 更 加 简洁 、 方 便 ， 下 面 介绍 一 种 通过 | 
jQuery 技术 实现 的 复 选 框 。 | 
专家 解答 | 
通过 jQuery 特性 实现 复 选 框 的 功能 主要 是 应 用 jQuery 中 的 事件 ， 通 过 它们 获取 表单 中 的 
元 素 ， 根 据 元 素 名 称 进行 判断 ， 进 而 执行 不 同 的 操作 。 其 主要 应 用 的 事件 如 下 : | 
(1) S$(document).ready0 方 法 是 事件 模块 中 最 重要 的 一 个 函数 ， 它 极 大 地 提高 了 Web 响应 | 
速度 。$(document) 用 来 获取 整个 文档 对 象 ， 从 这 个 方法 的 名 称 来 理解 ， 就 是 获取 文档 就 绪 的 时 | 
候 。 该 方法 的 书写 格式 为 : | 
$(document) .ready(function() { 
/在 这 里 写 代码 
站 
可 以 简写 成 : 
$0.ready(function() { 
/在 这 里 写 代 码 
D; | 
390% \ 
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当 $0 不 带 参数 时 ， 默 认 的 参数 就 是 document， 所 以 $0 是 $(document) 的 简写 形式 。 
还 可 以 进一步 简写 成 : 
S$(function(O) { 
// 在 这 里 写 代码 
DE 
虽然 语法 可 以 更 短 一 些 ， 但 是 不 提倡 使 用 简写 的 方式 ， 因 为 较 长 的 代码 更 具 可 读 性 ， 同 时 
| 也 可 以 防止 与 其 他 方法 混淆 。 
| 在 jQuery 中 可 以 使 用 $(document).ready0 方 法 代替 传统 的 window.onload() 方 法 ， 不 过 两 者 
| 之 间 还 是 存在 一 些 细 微 的 区 别 ， 主 要 表现 在 以 下 两 个 方面 。 
在 一 个 页 面 上 可 以 无 限制 地 使 用 $(documenb ready() 方 法 ， 各 个 方法 间 并 不 冲突 ， 会 
按照 在 代码 中 的 顺序 依次 执行 , 而 在 一 个 页 面 中 只 能 使 用 一 个 window.onload() 方 法 。 
在 一 个 文档 完全 下 载 到 浏览 器 时 〈 包 括 所 有 关联 的 文件 ， 如 图 片 、 横 幅 等 ) 就 会 响 
应 window.onload() 方 法 。 而 $(document)ready() 方 法 是 在 所 有 的 DOM 元 素 完全 就 绪 
以 后 就 可 以 调用 , 不 包括 关联 的 文件 。 如 在 页 面 上 还 有 图 片 没 有 加 载 完毕 但 是 DOM 
元 素 已 经 完全 就 绪 ， 这 样 就 会 执行 $S(documentbready0 方 法 ， 在 相同 条 件 下 
window.onload() 方 法 是 不 会 执行 的 ， 它 会 继续 等 待 图 片 加 载 ， 直 到 图 片 及 其 他 的 关联 
文件 都 下 载 完毕 时 才 执 行 。 所 以 说 $(document).ready(0 方 法 优 于 window.onload() 方 法 。 
| (2) 在 完成 页 面 的 加 载 之 后 ， 就 需 通过 事件 调用 不 同 的 方法 实现 不 同 的 功能 ， 这 里 应 用 
| 的 是 chickO 事 件 ， 通 过 它 调用 一 个 函数 ， 执 行 一 个 操作 。 
| 应 用 示例 
通过 jQuery 特效 实现 复 选 框 的 功能 ， 其 关键 代码 如 下 : 
<SCRIPT LANGUAGE="JavaScript"> 
$("document").ready(functionO{ 
$C"#all").click(functionO{ 
if(this.checked){ 
$("input[name='checkbox[]']").each(function| {this.checked=true;}); 
$("#btn1").attr("value"," 反 选 "); 
}else{ 
$("input[name='checkbox[]']").each(function(| {this.checked=false;}); 
$("#btn1").attr("value"," 全 选 "); 
b 
D); 
$("#btn1") .click(functionO{ 
$("[name='checkbox[]']").attr("checked",'true’): // 全 选 


D 
S$("#btn2").click(functionO{ 
$("[name='checkbox[]']").removeAttr("checked"): // 取 消 全 选 
D 
$("#btn3").click(functionO{ 
S$("[name='checkbox[]']:even").attr("checked",'true’): // 选 中 所 有 奇数 


出 
$("#btn4") clickGfunction0{ 
$("[name='checkbox[]']").each(functionO{ 
if($(this).attr("checked")){ 
S$(this).removeAttr("checked"): 


了 。330 。 
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}else{ 
S$(this).attr("checked",'true'"); 


在 jQuery 中 提供 了 很 多 的 事件 , 通过 它们 可 以 完成 很 多 操作 , 并 且 将 在 不 同 浏览 器 中 的 同 
-操作 方法 进行 了 统一 ， 使 其 功能 可 以 在 不 同 的 浏览 器 下 正常 运行 。jQuery 中 的 事件 如 表 10.3 


所 示 。 


方 ”法 
blurl 


blur(fn) 


表 10.3 jQuery 中 的 事件 


说 阴 
触发 元 素 的 blur 事 件 
在 每 一 个 匹配 元 素 的 blur 事 件 中 绑 定 一 个 处 理 函 数 , 在 元 素 失 去 焦点 时 触发 , 既 可 以 是 鼠 


标 行为 ， 也 可 以 是 使 用 Tab 键 离开 的 行为 


change0 触发 元 素 的 change 事 件 


在 每 一 个 匹配 元 素 的 change 事 件 中 绑 定 一 个 处 理 函 数 , 在 元 素 的 值 改变 并 失去 焦点 时 
change(fn) 触发 
chickO 触发 元 素 的 chick 事 件 
click(fn) 在 每 一 个 匹配 元 素 的 click 事 件 中 绑 定 一 个 处 理 函数 ， 在 元 素 上 单 击 时 触发 
dblclickO 触发 元 素 的 dblclick 事 件 
dblclick(fh 在 每 一 个 匹配 元 素 的 dblclick 事 件 中 绑 定 一 个 处 理 函 数 ， 在 某 个 元 素 上 双击 触发 


eITOl 
error(fn) 
focusO) 


focus(fn) 


在 每 
事件 


在 每 


触发 元 素 的 error 事 件 


-个 匹配 元 素 的 error 事 件 中 绑 定 一 个 处 理 函 数 , 当 JavaSprict 发 生 错误 时 , 会 触发 error 


触发 元 素 的 focus 事 件 
-个 匹配 元 素 的 focus 事 件 中 绑 定 一 个 处 理 函数 ， 当 匹配 的 元 素 获 得 焦点 时 触发 ， 通 
过 鼠标 单 击 或 按 Tab 键 触发 


keydownO 触发 元 素 的 keydown 事 件 


keydown(fn) “| 在 每 一 个 匹配 元 素 的 keydown 事 件 中 绑 定 一 个 处 理 函 数 ， 当 键盘 按 下 时 触发 

keyupO 触发 元 素 的 keyup 事 件 

keyup(fn) 在 每 一 个 匹配 元 素 的 keyup 事 件 中 绑 定 一 个 处 理 函 数 ， 会 在 按键 释放 时 触发 

keypressO 触发 元 素 的 keypress 事 件 

jaye 在 每 一 个 匹配 元 素 的 keypress 事 件 中 绑 定 一 个 处 理 函 数 ， 按 下 按键 时 触发 ( 即 按 下 并 抬 起 
同一 个 按键 7 

load(fn) 在 每 一 个 匹配 元 素 的 load 事 件 中 绑 定 一 个 处 理 函 数 ,匹配 的 元 素 内 容 完全 加 载 完毕 后 触发 

mousedown( 徊 )| 在 每 一 个 匹配 元 素 的 mousedown 事 件 中 绑 定 一 个 处 理 函 数 ， 鼠 标 在 元 素 上 单 击 后 触发 

mousemove(fn)| 在 每 一 个 匹配 元 素 的 mousemove 事 件 中 绑 定 一 个 处 理 函 数 ， 鼠 标 在 元 素 上 移动 时 触发 

mouseout(fn) ”| 在 每 一 个 匹配 元 素 的 mouseout 事 件 中 绑 定 一 个 处 理 函 数 ， 鼠 标 从 元 素 上 离开 时 触发 

mouseover(fn) | 在 每 一 个 匹配 元 素 的 mouseover 事 件 中 绑 定 一 个 处 理 函 数 ， 鼠 标 移入 对 象 时 触发 
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方法 说 明 
mouseup(fn) “| 在 每 一 个 匹配 元 素 的 mouseup 事 件 中 绑 定 一 个 处 理 函 数 ， 鼠 标 单 击 对 象 释放 时 触发 


resize(fn) 在 每 一 个 匹配 元 素 的 resize 事 件 中 绑 定 一 个 处 理 函 数 ， 当 文档 窗口 改变 大 小 时 触发 
scroll(fn) 在 每 一 个 匹配 元 素 的 scrol 事 件 中 绑 定 一 个 处 理 函 数 ， 当 滚动 条 发 生变 化 时 触发 

selectO 触发 元 素 的 select 事 件 

在 每 一 个 匹配 元 素 的 select 事 件 中 绑 定 一 个 处 理 函 数 ， 当 用 户 在 文本 框 〈 包 括 input 和 
textarea) 选中 某 段 文本 时 触发 

submitO 触发 元 素 的 submit 事 件 

submit(fn) 在 每 一 个 匹配 元 素 的 submit 事 件 中 绑 定 一 个 处 理 函 数 ， 表 单 提交 时 触发 

unload(fn) 在 每 一 个 匹配 元 素 的 unload 事 件 中 绑 定 一 个 处 理 函 数 ， 在 元 素 卸 载 时 触发 


select(fn) 


以 上 都 是 对 应 的 jQuery 事件 ， 和 传统 的 JavaScript 中 的 事件 几乎 相同 ， 只 是 名 称 不 同 。 方 
法 中 的 血 参 数 表示 一 个 函数 ， 事 件 处 理 程序 就 写 在 该 函数 中 。 


问题 221 如何 使 用 jQuery 技术 制作 级 联 羔 单 ? 


问题 阐述 

使 用 Ajax 实现 与 服务 器 端 进 行 交互 ， 代 码 编写 相对 烦琐 ,为 了 简化 Ajax 操作 ， 很 多 组 织 
都 推出 了 基于 Ajax 的 框架 ，jQuery 就 是 其 中 代表 之 一 。 它 是 继 prototype 之 后 又 一 个 优秀 的 
JavaScript 框架 ， 其 宗旨 是 写 更 少 的 代码 ， 做 更 多 的 事情 。 那 么 ， 如 何 使 用 jQuery 技术 制作 级 
联 菜单 呢 ? 
专家 解答 

jQuery 实现 js 代码 和 html 内 容 分 离 , 不 必 在 html 中 插入 一 堆 js 来 调用 命令 , 只 需 定义 id 
即 可 。 下 面 来 了 解 jQuery 中 基本 函数 的 应 用 以 及 其 代表 的 意义 。 

(1) $0， 用 于 替代 document.getElementById()。 其 代码 如 下 : 
Var someElement = $("#myId"):; 


通过 jQuery 获取 DOM 对象 中 的 元 素 ， 示 例 代码 如 下 : 


$("div m"); I 
$("div.admin"): 1 
$("div #mysqlid "): 1 
$("table a",content); I 


第 1 行 代码 获取 所 有 标签 下 的 m 元 素 ; 第 2 行 代码 获取 class 为 admin 的 元 素 ; 第 3 行 代码 
获取 标签 下 面 id 为 mysqlid 的 元 素 ; 第 4 行 代码 获取 content 为 上 下 文 的 table 中 所 有 的 连接 元 素 。 
(2) ready(fn)， 当 DOM 载 入 就 绪 可 以 查询 及 操纵 时 绑 定 一 个 要 执行 的 函数 。 该 方法 是 
window.load 事件 的 蔡 代 方法 。 
ready(fn) 
参数 血 (Function) 是 在 DOM 就 绪 时 执行 的 函数 ， 可 以 为 该 参数 任意 起 一 个 名 字 ， 并 因此 
32. 
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可 以 不 再 担心 命名 冲突 而 放心 地 使 用 $ 别 名 。 
注意 : | 
在 使 用 ready(fn) 时 必须 确保 在 <body> 元 素 的 onload 事件 中 没有 注册 函数 ,否则 将 不 会 触发 | 
$(document) ready(0) 事 件 。 另 外 ， 可 以 在 同一 个 页 面 中 无 限 次 地 使 用 $(document) ready0 事 件 。 | 
其 中 注册 的 函数 会 按照 代码 中 的 先后 顺序 依次 执行 。 


通过 $(document) ready(fn) 代 替 body 标签 的 onload 事件 ， 应 用 示例 如 下 : 
$(document) Teady(function0{ 
alert("jQuery"); 
D); | 
使 用 $(document).ready() 的 简写 ， 同时 内 部 的 jQuery 代码 依然 使 用 $ 作 为 别名 ， 而 不 必 考 虑 | 
全 局 的 $ 是 什么 。 其 jQuery 代码 如 下 : | 
jQuery(function($) { 
/可 以 在 这 里 继续 使 用 $ 作 为 别名 … 
D); 
上 述 $(document).ready() 的 简写 替换 的 是 如 下 代码 : 
<body onload="alert(jQuery"):"> | 
(3) jQuery 中 的 事件 机 制 。change0 触 发 每 个 匹配 元 素 的 change 事件 。 该 函数 会 调用 执 | 
行 绑 定 到 change 事件 的 所 有 函数 ， 包 括 浏览 器 的 默认 行为 。 可 以 通过 在 某 个 绑 定 的 函数 中 返 | 
回 false 来 防止 触发 浏览 器 的 默认 行为 。change 事件 会 在 元 素 失 去 焦点 时 触发 ， 也 会 当 其 值 在 | 
获得 焦点 后 改变 时 触发 。 | 
change(fn) 在 每 一 个 匹配 元 素 的 change 事件 中 绑 定 一 个 处 理 函数 。change 事件 会 在 元 素 失 | 
去 焦点 时 触发 ， 也 会 当 其 值 在 获得 焦点 后 改变 时 触发 。 | 
参数 f (Function) 是 在 每 一 个 匹配 元 素 的 change 事件 中 绑 定 的 处 理 函 数 。 
应 用 示例 1: 为 所 有 的 文本 框 增加 输入 验证 ， 代 码 如 下 : 
S$("input[type='text']").change( function() { 
/这 里 可 以 写 些 验证 代码 
让 | 
dlick(fn), 在 每 一 个 匹配 元 素 的 lick 事件 中 绑 定 一 个 处 理 函数 。 参 数 各 (Function) 是 绑 定 到 | 
click 事件 的 函数 。 | 
应 用 示例 2: 将 页 面 内 所 有 段落 点 击 后 隐藏 。 其 代码 如 下 : 


S$(document) .ready(functionO{ 
$("p").click( function (| { 
S(this).hideO; 
D; 
D 
(4) jQuery 中 的 Ajax 请 求 。 通 过 远程 HTTP GET 请 求 载 入 信息 。 语 法 如 下 : 
jQuery.get(url[data],[callback]) 


这 是 一 个 简单 的 GET 请 求 功能 ， 以 取代 复杂 S.ajax。 请 求 成 功 时 可 调用 回调 函数 。 如 果 需 | 
. 333 . 呈 
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元 
| 要 在 出 错时 执行 函数 ， 则 使 用 .ajax。 


| GET 请 求 返回 一 个 XMLHttpRequest。 参 数 url(String) 是 待 载 入 页 面 的 URL 地 址 ; data(Map) 
| 为 可 选 参数 ， 是 待 发 送 Key/value 参数 ，callback(Function) 也 是 可 选 参 数 ， 是 载 入 成 功 时 的 回调 


函数 。 
医 网 示例 1， 请 求 return php 网 页 ， 忽 略 返回 值 。 
$.get("return.php"); 
| 示例 2。 请 求 retum.php 网 页 ， 传 送 两 个 参数 ， 忽 略 返 回 值 。 
| $.get("return .php", { name: "php", time: "2pnm" } ); 
示例 3: 显示 return.php 返回 值 (HTML 或 XML， 取 决 于 返回 值 )。 


$.get("return.php", function(data){ 
alert("Data Loaded: " + data); 

Ds 
示例 4: 显示 retum.php， 传 递 两 个 参数 ， 添 加 返回 值 。 


$.get("return.php", {flag: "e",p: $("#p").val|)}, function(data){ 
$("#c").css("display","™"); 


$("#c").emptyO; 
S$("#c").append(data); /将 数据 追加 到 城市 下 拉 列表 
$("#cs").css("display","none"); // 初 始 状态 使 城市 下 拉 列 表 不 可 见 


)) 
通过 远程 HITP POST 请 求 载 入 信息 。 
| 这 是 一 个 简单 的 POST 请 求 功能 ， 以 取代 复杂 $.ajax。 请 求 成 功 时 可 调用 回调 函数 。 如 果 
| 需要 在 出 错时 执行 函数 ， 则 使 用 $.ajax。 
| 其 返回 值 和 参数 的 设置 与 HITP GET 相同 ， 应 用 方法 可 以 参考 GET 请 求 中 的 内 容 。 
| (5) jQuery 中 的 CSS。css(name)， 访 问 第 一 个 匹配 元 素 的 样式 属性 。 返 回 值 为 string， 参 
| 数 name 为 要 访问 的 属性 名 称 。 
| 示例 1: 取得 第 一 个 段落 的 color 样式 属性 的 值 。 
$("p").css("color"); 
css(name,value)， 在 所 有 匹配 的 元 素 中 设置 一 个 样式 属性 的 值 。 数 字 将 自动 转化 为 像素 值 。 
参数 name 为 属性 名 ， 参 数 value (String, Number) 是 属性 值 。 
示例 2: 初始 状态 下 隐藏 下 拉 列 表 。 


$("#cs").css("display","none"); // 初 始 状 态 使 城市 下 拉 列 表 不 可 见 
| (6) jQuery 中 的 文档 处 理 方法 。empty0， 删 除 匹 配 的 元 素 集合 中 所 有 的 子 节点 。 其 返回 
| 值 是 jQuery。 
| 示例 3: 清空 下 拉 列 表 中 的 子 元 素 。 

$("#c").empty(); 


| append(content) ， 向 每 个 匹配 的 元 素 内 部 追加 内 容 。 该 操作 与 对 指定 的 元 素 执 行 
| appendChild0) 方 法 ， 将 它们 添加 到 文档 中 的 情况 类 似 。 
其 返回 值 为 jQuery， 参 数 content (String, Element jQuery) 是 要 追加 到 目标 中 的 内 容 。 
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示例 4: 向 所 有 段落 中 追加 一 些 HIML 标记 。 
<p>Iwould like to say: </p> 
$("p").append("<b> Welcome to beijing</b>"); | 
制作 级 联 菜单 的 原理 是 : 通过 jQuery 中 的 get 发 送 请 求 ， 通过 append(contenb 向 下 拉 列 表 
中 添加 数据 通过 empty0 清 空 下 拉 列 表 ; 通过 css(name.value) 完 成 下 拉 列 表 的 隐藏 ， 通 过 ， 
change(fn) 事 件 实现 下 拉 列 表 的 关联 操作 。 其 关键 代码 如 下 : 
$(document) ready( | 
functionO{ | 
$.get("returnpc.php?flag=p", null function(data){// 向 服务 器 发 送 GET 请 求 ， 获 取 省 份 的 值 | 
$("#p").append(data); // 将 结果 追加 到 省 份 下 拉 列 表 中 | 


JE 
$("#c").css("display","none"); // 初 始 状态 使 城市 下 拉 列 表 不 可 见 
$("#p").change(functionO{ // 为 省 份 下 拉 列 表 增 加 改变 事件 
if($("#p").valO—""){ // 在 没 选择 省 份 的 情况 下 ， 使 城市 下 拉 列 表 不 可 见 
S$("#c").css("display","none"): 
}else{ 
$.get("retumnpc.php?flag=cé&p="+$("#p").val(), null, function(data){ 
// 如 果 选 择 了 某 省 份 发 送 GET 请 求 
$("#c").css("display",""); 。“// 并 使 城市 下 拉 列 表 可 见 
S$("#c").emptyO; 
$("#c").append(data); // 将 数据 追加 到 城市 下 拉 列 表 
$("#cs").css("display","none");， ”// 初 始 状态 使 城市 下 拉 列 表 不 可 见 
D); 
): : 
$("#cs").css("display","none"); // 初 始 状 态 使 城市 下 拉 列 表 不 可 见 
S$("#c").change(finctionO){ 
这 SO#c val0 一 "){ 
S$("#cs").css("display","none"); 
}else{ 


S$.get("returnpe.php?flag=cs&c="+$("#c").valO+"&p="+$("#p").val|,null,function(data) { 
SC"#es") .css("display™,"™"): 
S$("#cs").emptyO; 
S$("#cs").append(data): // 将 数据 追加 到 城市 下 拉 列 表 | 
)); | 


应 用 示例 | 
本 示例 中 应 用 jQuery 制作 一 个 级 联 菜单 ， 实 现 省 、 市 和 县 的 3 级 联动 ， 其 具体 步骤 如 下 : | 
(1) 创建 ndex.php 文件 ， 完 成 主页 面 的 设计 。 首 先 通过 script 标签 调用 jQuery 框架 包 文 | 
件 jquery-1.3.2jjs 以 及 funjjs 文件 。 然后， 通过 <select></selec 亿 标签 ,创建 3 个 下 拉 列 表 ， 用 于 
输出 省 、 市 和 县 的 内 容 ， 分 别 定义 它们 的 i 为 p、c 和 cs。 其 关键 代码 如 下 : 
<script language="javascript" src="js/jquery-1.3.2.js"></script> 
<script language="javascript" src="js/fun.js"></script> 
335 
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<td height="50" colspan="2" align="center" valign="middle"> 查 询 全 国 各 个 城市 的 天 气 情况 : 
<select id="p" name="p" > 
<option value="">- 请 选择 地 区 -</option> 


# | </select> 
食 一 | | <select id="c" name="c"> </select> 
| <select id="cs" name="cs"> </select> 
| </tr> 


| (2) 创建 jQuery 中 GET 请 求 的 retumpc.php 文件 ,在 GET 请 求 中 传递 的 是 省 、 市 和 县 的 
| id。 首先 ， 通 过 require_ once 调用 存储 省 、 市 和 县 数据 的 文件 pcphp， 获 取 GET 传递 的 参数 ， 
| 定义 变量 sr。 代 码 如 下 : 


<2php 

require_once 'pe.php'; // 包 含 省 份 数组 和 城市 数组 

$flag = $_GET['flag’]; // 用 于 区 分 是 返回 省 份 信息 还 是 返回 城市 信息 
gst 


| 然后 ， 判 断 当 参 数值 为 p 时 ， 将 数组 p 中 存储 的 省 份 信息 添加 到 省 份 下 拉 列 表 中 。 将 数据 
| 赋 给 变量 sr， 并 输出 变量 str。 代 码 如 下 : 


if($flag 一 D){ // 如 果 参 数 flag 的 值 为 p， 则 返回 省 份 信息 
for($i=0; $i<count($p): $i++){ // 通 过 循环 整理 所 有 要 追加 到 省 份 下 拉 列 表 的 内 容 
$str.="<option 


| value=\"" .iconv('gbk','utf-8',$p[$i])."\">".iconv('gbk",'utf-8',$p[$1])."</option>"; 
| } 
echo $str; // 输 出 要 返回 的 参数 


| 接着 ,判断 当 参 数值 为 c 时 ， 表 示 返 回 的 是 市 的 信息 ， 提 取 参 数值 p 所 对 应 的 省 份 在 省 份 

| 数组 中 的 键 值 ， 根 据 省 份 的 键 值 确定 所 对 应 的 市 ， 并 通过 for 循环 将 市 数组 中 的 数据 添加 到 

| 市 下 拉 列 表 中 。 将 数据 赋 给 变量 sttr， 并 输出 变量 str。 代码 如 下 : 

| }elseif ($flag—c){ // 如 果 flag 的 值 等 于 ce， 表示 返回 的 是 市 级 城市 
$index = array_search($_GET['p"], $p); // 提 取 所 选择 省 份 的 在 省 份 数 组 中 对 应 的 键 值 

for($j=0; $j<count($c[$index]); $j++){ // 通 过 循环 整理 所 有 要 追加 到 城市 下 拉 列 表 的 内 容 
$str= "<option value=\" iconv(‘gbk','utf-8',$c[Sindex][$j])."\">" iconv('gbk’,'utf-8,, $c[Sindex][$j]). 


| "</option>"; 
| } 
echo $str; // 输 出 要 返回 的 参数 


| 最 后 ， 判 断 当 参数 值 为 cs 时 ， 表 示 返 回 的 是 县 的 信息 ， 分 别提 取 参 数值 p 所 对 应 的 省 份 
| 在 省 份 数组 中 的 键 值 和 参数 值 c 所 对 应 的 市 在 市 数组 中 的 键 值 ， 根 据 省 份 的 键 值 和 市 的 键 值 确 
| 定 所 对 应 的 县 ， 最 终 通过 for 循环 将 县 数组 cs 中 的 数据 添加 到 县 下 拉 列 表 中 。 将 数据 赋 给 变量 
str， 并 输出 变量 sr。 代 码 如 下 : 
| Jelset 
Sindex = array_search($_GET['p'], $p); // 提 取 所 选择 省 份 的 在 省 份 数 组 中 对 应 的 键 值 
$indexs = array_search($_GET['c], $c[$index]);// 提 取 所 选择 省 份 的 在 市 城市 数组 中 对 应 的 键 值 
for($k=0; $k<count($cs[$index][$indexs]); Sk++){ ”// 通 过 循环 整理 所 有 要 追加 到 城市 下 拉 
列表 的 内 容 
$str= "<option value=\"".iconv(‘gbk','utf-8',$cs[$index][$indexs][$k])."\">".iconv('gbk’,"utf-8', 
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$cs[$index][$indexs][$k])."</option>"; 
} 


echo $str; // 输 出 要 返回 的 参数 
1 
四 | 动 
(3) 创建 pephp 文件 ， 在 该 文件 中 包含 p、c 和 cs 3 个 数组 ， 分 别 用 于 存储 省 、 市 和 县 的 
数据 。 运 行 本 示例 ， 运 行 结果 如 图 10.9 所 示 。 Np 
ee | 首页 | 天 气 预 报 | 天 气 趋势 | 气象 新 闻 | 旅游 气象 台 | 城市 天 气 预 报 | 
WWw.mrbccd.com 
碍 齐全 国 各 个 城市 的 天 气 情况 : [河北 EIEERITECE 已 | 
和 
Copyright © 2008 wr. arbecd com, VeatherOnline hsis Ltd 各 1 Rights Reserved. 
图 10.9 jQuery 制作 级 联 菜单 
专家 点 评 


要 应 用 jQuery 技术 ， 其 前 提 是 必须 在 程序 中 包含 jQuery 的 框架 包 。 本 示例 中 使 用 的 是 | 
jquery-1.3.2Jjs。 | 


问题 222 ”如 何 通过 jQuery 生成 网 页 特效 ? 


问题 阐述 | 

在 JavaScript 中 ， 生 成 网 页 特效 编写 的 代码 往往 非常 繁琐 、 复 杂 ， 而 通过 jQuery 来 生成 网 | 
页 的 特效 却 非常 简单 ， 而 且 代码 也 非常 少 。 那 么 ， 它 具体 的 操作 步骤 是 怎样 的 呢 ? 又 应 用 到 哪 | 
些 技术 呢 ? | 


专家 解答 
通过 jQuery 生成 网 页 特效 主要 应 用 的 是 jQuery 中 的 特效 函数 ， 下 面 对 jQuery 中 的 常用 特 
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效 函 数 进 行 讲解 。 
(1) 基本 效果 ， 实 现 匹 配 元 素 的 隐藏 和 显示 。hide0 隐 藏 显示 的 元 素 。 它 是 hide( speed. 
[callback] ) 的 无 动画 版 。 如 果 选 择 的 元 素 是 隐藏 的 ， 则 该 方法 将 不 会 改变 任何 东西 。 
示例 : 隐藏 所 有 段落 。 
$CP'")hideO 

hide(speed,[callback]) 以 优雅 的 动画 隐藏 所 有 匹配 的 元 素 ， 并 在 显示 完成 后 有 选择 地 触发 一 
个 回调 函数 ， 同 时 可 以 根据 指定 的 速度 动态 地 改变 每 个 匹配 元 素 的 高 度 、 宽 度 和 不 透明 度 。 

参数 speed(String,Number) 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 者 
表示 动画 时 长 的 毫秒 数值 (如 1000)。 

参数 callback(FunctionFunction) : (Optional) 表 示 在 动画 完成 时 执行 的 函数 , 每 个 元 素 执行 
一 次 。 

示例 1: 用 600 毫秒 的 时 间 将 段落 缓慢 隐藏 。 

$("p").hide("slow"); 

示例 2: 用 200 毫秒 将 段落 迅速 隐藏 ， 并 弹出 一 个 对 话 框 。 

$C"p") hide("fast",functionO{ 
alert("Welcome To BEI JING."); 
D); 

show0 显 示 隐 藏 的 匹配 元 素 。 

它 是 show( speed, [callback] ) 无 动画 的 版 本 。 如 果 选 择 的 元 素 是 可 见 的 , 则 该 方法 将 不 会 改 
变 任 何 东 西 。 无 论 这 个 元 素 是 通过 hide() 方 法 隐藏 的 ， 还 是 在 CSS 中 设置 了 display:none:， 该 
方法 都 将 有 效 。 

示例 : 显示 所 有 段落 。 

<p style="display: none">PHP</p> 
$("p").showO 

show(speed,[callback]) 以 优雅 的 动画 显示 所 有 匹配 的 元 素 , 并 在 显示 完成 后 有 选择 地 触发 一 
个 回调 函数 ， 同 时 可 以 根据 指定 的 速度 动态 地 改变 每 个 匹配 元 素 的 高 度 、 宽 度 和 不 透明 度 。 

参数 speed (String,Number) 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 者 
表示 动画 时 长 的 毫秒 数值 (如 1000)。 

参数 callback (Function) : (Optional) 表 示 在 动画 完成 时 执行 的 函数 ， 每 个 元 素 执行 一 次 。 

示例 1: 用 缓慢 的 动画 将 隐藏 的 段落 显示 出 来 ， 用 时 600 毫秒 。 代 码 如 下 : 

<p style="display: none">PHP</p> 
S$("p").show("slow"); 
示例 2: 迅速 将 隐藏 的 段落 显示 出 来 ， 用 时 200 上 毫秒。 执行 后 调用 回调 函数 。 代 码 如 下 : 


<p style="display: none">Hello</p> 
$("p").show("fast",function|){ 
S$(this).text("Welecome To BEIJING!"); 
D; 


示例 3: 将 隐藏 的 段落 用 将 近 4 秒 的 时 间 显示 出 来 ， 执 行 后 调用 回调 函数 。 代 码 如 下 : 
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<p style="display: none">Hello</p> 
$("p").show(4000,functionO{ 
S$(this).text("Welcome To BeiJing..."): 
D; 
toggle0 切 换 元 素 的 可 见 状态 。 如 果 元 素 是 可 见 的 ， 则 切换 为 隐藏 的 ， 如 果 元 素 是 隐藏 的 ， 
则 切换 为 可 见 的 。 

示例 : 切换 所 有 段落 的 可 见 状态 。 代 码 如 下 : 

<p>PHP</p><p style="display: none">Welcome To BeiJing! </p> 

$("p").toggleO| 


出 结果 : <p tyle="display: none">PHP</p><p style="display: block"> Welcome To BeiJing! | 


苏 


</p>。 
(2) 淡 入 、 淡 出 效果 。 fadeln(speed [callback]) 通 过 不 和 明度 的 变化 来 实现 所 有 匹配 元 素 的 
淡 入 效果 ， 并 在 动画 完成 后 有 选择 地 触发 一 个 回调 函数 。 
这 个 动画 只 调整 元 素 的 不 透明 度 , 也 就 是 说 , 所 有 匹配 的 元 素 的 高 度 和 宽度 不 会 发 生变 化 。 | 
参数 speed (String,Number) 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 者 | 
表示 动画 时 长 的 毫秒 数值 (如 1000)。 | 
参数 callback (Function) : (Optional) 为 可 选 参数 ， 表 示 在 动画 完成 时 执行 的 函数 。 
示例 1: 用 600 毫秒 缓慢 将 段落 淡 入 。 代 码 如 下 : 
$("p").fadeIn("slow"): 


示例 2: 用 200 毫秒 快速 将 段落 淡 入 ， 并 弹出 一 个 对 话 框 。 代 码 如 下 : 
$C'p") fadeIn("fast",fanctionO{ 
alert("Welcome To BeiJing."); 
»); | 
参数 fadeOut(speed,[callback]) 通 过 不 透明 度 的 变化 来 实现 所 有 匹配 元 素 的 淡出 效果 ， 并 在 | 
动画 完成 后 有 选择 地 触发 一 个 回调 函数 。 
这 个 动画 只 调整 元 素 的 不 透明 度 , 也 就 是 说 , 所 有 匹配 的 元 素 的 高 度 和 宽度 不 会 发 生变 化 。 
参数 speed (String,Number) 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 者 | 
表示 动画 时 长 的 毫秒 数值 (如 1000)。 | 
参数 callback (Function) 为 可 选 参数 ， 表 示 在 动画 完成 时 执行 的 函数 
示例 1: 用 600 毫秒 缓慢 地 将 段落 淡出 。 代 码 如 下 : 
$C'p") fadeOut("slow"): 
示例 2: 用 200 毫秒 迅速 地 将 段落 淡出 ， 并 弹出 一 个 对 话 框 。 代 码 如 下 : 
S$('p") fadeOut("fast",fimetionO{ 
alert("Welcome To BeiJing."); 
D; | 
fadeTo(speed,opacity,[callback]) 把 所 有 匹配 元 素 的 不 透明 度 以 渐进 方式 调整 到 指定 的 不 透 | | 
明度 ， 并 在 动画 完成 后 有 选择 地 触发 一 个 回调 函数 。 | 
这 个 动画 只 调整 元 素 的 不 透明 度 ， 也 就 是 说 ， 所 有 匹配 元 素 的 高 度 和 宽度 不 会 发 生变 化 。 | 
参数 speed (String,Number) 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 者 | 
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| 表示 动画 时 长 的 毫秒 数值 (如 1000)。 

| 参数 opacity (NumbeD 表 示 要 调整 到 的 不 透明 度 值 (0~1 之 间 的 数字 )。 

| 参数 callback (Function) 为 可 选 参数 ， 为 在 动画 完成 时 执行 的 函数 。 
会 内 | 示例， 用 600 毫秒 缓慢 地 将 段落 的 透明 度 调整 到 0.66， 大 约 23 的 可 见 度 。 代 码 如 下 : 
Se $("p").fadeTo("slow", 0.66);$("p").fadeTo("slow", 0.66); 


(3) 滑动 效果 。slideDown(speed.[callback]) 通 过 高 度 变化 (向 下 增 大 ) 来 动态 地 显示 所 有 
| 匹配 的 元 素 ， 在 显示 完成 后 可 以 有 选择 地 触发 一 个 回调 函数 。 
这 个 动画 效果 只 调整 元 素 的 高 度 ， 可 以 使 匹配 的 元 素 以 “滑动 ”的 方式 显示 出 来 。 
| 参数 speed (String,Number) 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 者 
| 表示 动画 时 长 的 毫秒 数值 (如 1000)。 
| 参数 callback (Function) 为 可 选 参数 ， 为 在 动画 完成 时 执行 的 函数 。 
示例 1: 用 600 毫秒 缓慢 的 将 段落 滑 下 。 代 码 如 下 : 
$("p'").slideDown("slow"); 
示例 2: 用 200 毫秒 迅速 地 将 段落 滑 下 ， 并 弹出 一 个 对 话 框 。 代 码 如 下 : 
S$('p") slideDown("fast",functionO{ 
alert("Welcome To BeiJing."); 
D); 
| slideToggle(speed,[callback]) 通 过 高 度 变化 来 切换 所 有 匹配 元 素 的 可 见 性 ， 并 在 切换 完成 后 
| 有 选择 地 触发 一 个 回调 函数 。 
| 这 个 动画 效果 只 调整 元 素 的 高 度 ， 可 以 使 匹配 的 元 素 以 “滑动 ”的 方式 隐藏 或 显示 。 
| 参数 speed (String.NumbenD 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 表 
| 示 动 画 时 长 的 毫秒 数值 (如 1000)。 
参数 callback (Function) 为 可 选 参数 ， 表 示 在 动画 完成 时 执行 的 函数 。 
示例 1: 用 600 毫秒 缓慢 地 将 段落 滑 上 或 滑 下 。 代 码 如 下 : 
$("p").slideToggle("slow"): 
示例 2: 用 200 毫秒 迅速 地 将 段落 滑 上 或 滑 下 ， 并 弹出 一 个 对 话 框 。 代 码 如 下 : 
$(0"p".slideToggle("fast".functionO{ 
alert("welecome To BeiJing."); 
D; 
| slideUp(speed,[callback]) 通 过 高 度 变化 (向 上 减 小 ) 来 动态 地 隐藏 所 有 匹配 的 元 素 , 在 隐藏 
| 完成 后 有 选择 地 触发 一 个 回调 函数 。 
| 这 个 动画 效果 只 调整 元 素 的 高 度 ， 可 以 使 匹配 的 元 素 以 “滑动 ”的 方式 隐藏 起 来 。 
| 参数 speed (String,Number) 表 示 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal" or "fast") 或 表 
| 示 动 面 时 长 的 毫秒 数值 (如 1000)。 
| 参数 callback (Function) 为 可 选 ， 表 示 在 动画 完成 时 执行 的 函数 。 
示例 1: 用 600 毫秒 缓慢 地 将 段落 滑 上 。 代 码 如 下 : 
S$("p").slideUp("slow"); 
示例 2: 用 200 毫秒 迅速 地 将 段落 滑 上 ， 并 弹出 一 个 对 话 框 。 代 码 如 下 : 
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$("p").slideUp("fast",functionO{ 
alert("Welecome To BeiJing."); 
DE 
应 用 示例 | 
本 示例 应 用 jQuery 实现 网 页 中 图 片 的 淡 入 和 淡出 的 效果 ， 并 且 实 现 数据 的 间断 性 滑动 更 | 
换 。 其 具体 实现 步骤 如 下 : 
(1) 创建 js 文件 夹 , 载 入 jQuery 框架 包 ; 创建 js 脚本 文件 , 编写 自 定义 函数 , 通过 jQuery | 
特效 函数 定义 网 页 特效 功能 。 在 本 示例 中 实现 3 种 效果 ， 图 片 的 淡 入 和 淡出 以 及 页 面 内 容 的 滑 | 
动 更 换 。jQueryjs 脚本 文件 的 关键 代码 如 下 : | 
finetion AOt // 创 建 自 定义 函数 


$("#a").fadeOut("slow",functionO{ /实现 图 片 的 淡出 效果 
$("#e").fadeIn("slow"): // 执 行 回调 函数 ， 实 现 另外 一 个 图 片 的 淡 入 


1 
$("#aa").fadeOut("slow"); // 执 行 图 片 的 淡出 操作 
$("#ce").fadeOut("slow"); 
$("#dd").fadeOut("slow"); 
$("#bb").fadeIn("slow"); 
} 
setTimeout("AO", 10800); /间隔 一 段 时 间 执行 一 次 函数 
function BO{ 
$("#b").fadeOut("slow",functionO{ 
S$("#f").fadeIn("slow"); 
1 
$("#aa") .fadeOut("slow"):; 
$("#bb").fadeOut("slow"); 
$("#dd").fadeOut("slow"); 
$("#ce") fadeIn("slow"): 
} 


(2) 创建 index.php 页 面 , 通过 Script 脚本 包含 js 脚本 文件 和 jQuery 框架 包 , 应 用 onClick 
事件 调用 自 定义 函数 ， 实 现 图 片 的 淡 入 和 淡出 以 及 内 容 的 替换 。 其 关键 代码 如 下 : 
<script sre="js/jquery-1.3.2.js"></script> 
<script sre="js/jquery.js"></script> 
<tr> 
<td width="450" align="right"> | 
<img src="images/dl 08-(9) 01.jpg" width="11" height="164" /> | 
<img src="images/dl 08-(9) 02.jpe” name="a" width="106" height="164" border="0" | 
usemap="#aMap" id="a" style="display:inline" onclick="AO:" > | 
<img src="images/dl 08-(9) 03.jpg" width="109" height="164" border="0" usemap="#bMap" 
id="b" style="display:inline" onclick="BO:;" /> | 
se 省 略 部 分 代码 | 
<img id="h" src="images/dl 08-(9) 05.jpg" width="113" height="164" style="display:none" | 
onclick="HO:;" border="0" /></td> ! 
<td width="350"><img src="images/dl 08(10)jpg" width="350" height="164" /></td> | 
</t> | 
<tr> | 
<td width="450" height="180" align="right" valign="top" id="aa" style="display:line" > | 
<table width="448" border="0" cellpadding="0" cellspacing="0"> | 
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| < 
| <td width="11" height="30">&nbsp:</td> 
| <td width="427" align="center"><img src="images/dl 082.jpg" width="401" 
| height="28" /></td> 
会 内 | <td width="10">&nbsp:</td> 
RA | </tr> 
| <tr> 
<td>&nbsp;</td> 
<td height="150" valign="top"> 
<table width="427" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="45" height="10">&nbsp:</td> 
<td width="167">&nbsp:</td> 
<td width="215">&nbsp:</td> 
</tr> 
<2php 


include_once("conn/conn.php"); 
Sresult=mysql_query("select * from tb_articles",$conn); 
while($myrow=mysql fetch array($result)){ 


pes 
<tr> 
<td height="22" align="right" class="STYLE3">.</td> 
<td align="left" class="STYLE3"><a href="#"><?php echo 
$myrow[topic];?></a></td> 
<td class="STYLE3"><?php echo $myrow[addtime];?></td> 
</tr> 
<2php 
} 
?> 
</table></td> 
<td>&nbsp;</td> 
</tr> 
</table></td> 
</tr> 
运行 结果 如 图 10.10 所 示 。 
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图 10.10 jQuery 生成 网 页 特效 
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和 有 划 理 理 理 理 理 吾 吾 


问题 223 
问题 224 
问题 225 
问题 226 
问题 227 
问题 228 
问题 229 
问题 230 
问题 231 


如 何 处 理 XML 中 的 数据 刷新 的 问题 ? 
RSS 2.0 都 包括 哪些 元 素 ? 

如 何 创 建 支持 RSS 阅读 的 站 点 ? 

如 何 动态 创建 XML 文件 ? 

如 何 动 态 读 取 XML 文件 ? 

如 何 查询 XML 文件 中 的 数据 ? 

如 何 动态 更 新 XML 中 的 数据 ? 

如 何 动态 删除 XML 中 的 数据 ? 

如 何 通过 jQuery 读 取 XML 中 的 数据 ? 


评 


需 
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问题 223 如何 处 理 XML 中 的 数据 刷新 的 问题 ? 


鲜 | ， 问 题 阐述 
在 调用 XML 文件 中 的 数据 时 ， 由 于 数据 经 常 更 新 ， 所 以 必须 刷新 XML 文件 才能 输出 最 
| 新 的 数据 。 但 是 在 刷新 XML 文件 的 过 程 中 ， 直 接 刷 新 页 面 不 能 对 XML 文件 中 的 数据 进行 更 
| 新 ， 而 必须 要 关闭 网 页 ， 再 重新 打开 ， 才 能 完成 XML 文件 中 数据 的 刷新 ， 操 作 起 来 非常 烦琐 。 


| 那么 ， 如 何 解决 XML 中 的 数据 刷新 的 问题 呢 ? En 
| 专家 解答 党 
解决 此 问题 的 方法 为 : ee 


(1) 在 浏览 器 中 选择 “工具 ”一 “Intemet 选项 ”命令 。 
(2) 选择 “常规 ”选项 卡 ， 在 该 页 面 中 单 击 “ 设 置 ” 


全 用 的 磅 条 字 间 人) 


| [mm 

| 按钮， 打开 “设置 ”窗口 。 

| 称 动 文件 灵 叫 .| 查看 文件 WW，，| 查看 对 象 由) ， 

| (3) 在 “设置 ”窗口 中 选中 “每 次 访问 此 页 时 检查 ” ee 
| 单 选 按 钮 ， 如 图 11.1 所 示 。 we 
| (4) 单 击 “确定 ”按钮 ， 完 成 设置 。 图 11.1 “设置 ”窗口 


问题 224 RSS 2.0 都 包括 哪些 元 素 ? 


| 问题 阐述 
| ”什么 是 RSS? RSS 2.0 都 包括 哪些 元 素 ? 
| 专家 解答 


| RSS 为 Really Simple Syndication( 简 易 供 稿 ) 的 缩写 ， 是 某 一 站 点 用 来 和 其 他 站 点 之 间 共 
| 享 内 容 的 一 种 简易 方式 , 也 叫 聚 合 内 容 。 网络 用 户 可 以 在 客户 端 借助 于 支持 RSS 的 新 闻 聚 合 工 
| 具 软 件 (如 SharpReader NewzCrawler、FeedDemon RSSReader)， 在 不 打开 网 站 内 容 页 面 的 情 
| 况 下 阅读 支持 RSS 输出 的 网 站 内 容 。 

| 从 RSS 阅读 者 的 角度 来 看 , 完全 不 必 考 虑 它 到 底 是 什么 意思 , 只 要 简单 地 理解 为 一 种 方便 
， 的 信息 获取 工具 就 可 以 了 。RSS 获取 信息 的 模式 与 加 入 邮件 列表 〈 如 电子 杂志 和 新 闻 邮 件 ) 获 
| 取信 息 有 一 定 的 相似 之 处 ， 也 就 是 可 以 不 必 登 录 各 个 提供 信息 的 网 站 而 通过 客户 端 浏 览 方式 

( 称 为 “RSS 阅读 器 ” 或 者 在 线 RSS 阅读 方式 浏览 内 容 。 
1，RSS 的 历史 

| RSS 最 早 的 版 本 是 0.90, 是 由 Netscape 公司 设计 的 , 目的 是 用 来 建立 一 个 整合 了 各 主要 新 
| 闻 站 点 内 容 的 门户 ， 但 是 0.90 版 本 的 RSS 规范 过 于 复杂 ， 随 后 简化 的 RSS 0.91 版 本 也 随 着 
| Netscape 公司 对 该 项 目的 放弃 而 于 2000 年 暂停 。 

| 后 来 ， 一 家 专门 从 事 博 客 写作 软件 开发 的 公司 UserLand 接手 了 RSS 0.91 版 本 的 发 展 ， 并 
| 把 它 作为 其 博客 写作 软件 的 基础 功能 之 一 继续 开发 ， 逐步 推出 了 0.92、0.93 和 0.94 版 本 。 随 着 
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网 络 博客 的 流行 ，RSS 作为 一 种 基本 的 功能 也 被 越 来 越 多 的 网 站 和 博客 软件 支持 。 
在 UserLand 公司 接手 并 不 断 开发 RSS 的 同时 ， 很 多 的 专业 人 士 意识 到 需要 通过 一 个 第 三 
方 、 非 商业 的 组 织 ， 把 RSS 发 展 成 为 一 个 通用 的 规范 ， 并 进一步 标准 化 。 于 是 2001 年 一 个 联 
合 小 组 在 0.90 版 本 RSS 的 开发 原则 下 ,以 W3C 新 一 代 的 语义 网 技术 RDF(Resource Description 
Framework) 为 基础 ， 对 RSS 进行 了 重新 定义 ， 发 布 了 RSS 1.0， 并 将 RSS 定义 为 “RDF Site 
Summary”。 但 是 这 项 工作 没有 与 UserLand 公司 进行 有 效 的 沟通 ，UserLand 公司 也 不 承认 RSS 
1.0 的 有 效 性 , 并 坚持 按照 自己 的 设想 进一步 开发 出 RSS 的 后 续 版 本 , 到 2002 年 9 月 发 布 了 最 
新 版 本 RSS 2.0，UserLand 公司 将 RSS 定义 为 “Really Simple Syndication”。 
到 目前 为 止 , RSS 从 0.91 起 到 2.01, 一 共有 7 个 版 本 之 多 。 各 个 版 本 除 1.0 以 外 大 同 小 异 。 
在 RSS 阅读 器 中 读 取 的 是 站 点 中 一 个 标准 格式 的 RSS 2.0 文件 , 所 以 说 要 让 站 点 支持 RSS 
阅读 ， 标 准 RSS 2.0 格式 文件 的 创建 是 关键 。RSS 2.0 结构 可 以 划分 为 3 个 部 分 。 
RSS 的 根 元 素 rss。 
回 ”在 根 元 素 下 的 channel 元 素 〈 表 示 新 闻 频 道 的 特性 ) 。 
在 根 元 素 下 channel 元 素 之 后 出 现 的 item 元 素 〈 表 示 新 闻 条 目 ) 。 
2，RSS 的 根 元 素 
rss 元 素 是 RSS 的 根 元 素 ， 具 有 version 属性 ， 可 以 包含 任何 名 称 空间 的 其 他 属性 。 在 该 元 
素 下 的 子 元 素 有 两 个 : channel (频道 ， 只 能 出 现 一 次 ) 和 item (条 目 ， 可 以 出 现 多 次 )。 
3. channel ( 频道 ) 元 素 及 其 子 元 素 
channel 元 素 表 示 网 站 上 所 发 布 的 新 闻 栏 目 ， 该 元 素 的 子 元 素 如 表 11.1 所 示 。 


表 11.1 channel 元 素 的 可 用 子 元 素 

元 素 名 称 说 了 明 
title 网 站 上 新 闻 频 道 的 标题 ， 该 标题 应 该 与 html 网 站 的 标题 保持 一 致 
link 可 以 到 达 此 频道 相关 的 HTML 网 页 的 超 链接 
description 频道 的 描述 
language 频道 对 应 的 语种 
copyright 网 站 著作 权 的 声明 
managingEditor | 频道 内 容 编辑 的 电子 邮件 地 址 
webMaster 维护 网 站 频道 的 主要 技术 人 员 


pubDate 发 布 RSS 的 时 间 

lastBuilDate “| 最 后 更 新 网 站 内 容 的 时 间 

category 该 栏目 所 属 的 类 别 ， 此 元 素 包 含 “domain ”属性 
generator 生成 RSS 栏 目的 程序 

docs 与 RSS 所 执行 的 版 本 标准 对 应 的 URL (很 少 用 ) 


表示 应 用 程序 注册 订阅 此 频道 的 信息 ，RSS 应 用 程序 使 用 此 元 素 的 信息 ， 自 动 检查 RSS 
更 新 。 此 元 素 的 子 元 素 如 下 : 
domain: 自动 下 载 RSS 时 的 域名 
cloud port: 下 载 RSS 时 所 采用 的 端口 
path: RSS 在 域名 下 的 路 径 
registerProcedure: 订阅 RSS 时 所 执行 的 程序 
protocol: 传输 下 载 RSS 时 所 用 的 网 络 协议 ， 如 HTTP 等 
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Fn py 
CS 
元 素 名 称 说 明 
世 RSS 内 容 在 客户 浏览 器 中 缓存 的 时 间 
image 与 频道 对 应 的 图 标 
可 以 用 来 接受 文本 的 输入 框 ， 子 元 素 如 下 : 
title: 提交 按钮 上 的 文字 
textInput description: 对 此 文本 框 的 描述 
name: 文本 框 的 名 称 
link: 服务 器 上 处 理 文本 请 求 的 应 用 程序 路 径 
RSS 下 载 程序 不 检查 RSS 更 新 的 时 间 ， 此 元 素 包含 hour 子 元 素 (最 多 24 个 ) ， 表 示 不 检 
skipHours ”| 查 RSS 更 新 的 钟点 ,hour 元 素 值 所 代表 的 每 个 时 间 对 应 格林 威 治 时 间 (GMT,，Greenwich 
Mean Time) 的 小 时 
RSS 下 载 程序 可 以 跳跃 不 检查 RSS 更 新 的 时 期 ， 此 元 素 包含 day 子 元 素 〈 最 多 7 个 ) ， 表 
skipDays ”| 示 不 检查 RSS 更 新 的 日 期 , day 元 素 的 值 分 别 用 Monday、Tuesday、Wednesday、 Thursday、 


Friday、Saturday 和 Sunday 表 示 星 期 一 到 星期 日 


4. item (项 目 ) 元 素 及 其 子 元 素 
item 元 素 表示 网 站 上 所 发 布 的 新 闻 ， 该 元 素 的 子 元 素 如 表 11.2 所 示 。 


元 素 名 称 
title 
link 


表 11.2 item 元 素 的 可 用 子 元 素 
说 了 明 
新 闻 标题 
阅读 新 闻 正 文 的 超 链接 


description _ | 新 闻 内 容 的 摘要 


author 


本 条 目 作 者 的 电子 邮件 地 址 


category 此 新 闻 所 属 的 类 别 ， 包 含 domain 属 性 〈 关 于 此 类 别 的 领域 名 称 ) 


comments ”| 此 新 闻 条 目 对 应 的 评论 网 页 的 超 链接 
ne 与 此 条 目 相 关 的 多 媒体 的 内 容 ， 此 元 素 包含 url (下 载 附 件 的 超 文本 链接 路 径 ) 、length 
(附件 字 节 大 小 ) 和 type《〈 附 件 的 MIME 媒 体 类 型 ) 属性 
1 全 球 唯一 标识 符 〈Globally Unique Identifier) 或 永久 的 URL 值 ， 此 元 素 包 含 isPermaLink 
人 (是 否 为 永久 的 URL 值 》 属 性 
pubDate 发 布 此 新 闻 的 时 间 
source 此 新 闻 的 来 源 ， 此 元 素 包含 url (此 新 闻 所 引 用 的 频道 的 路 径 ) 属性 
专家 点 评 


上 述 介绍 的 内 容 就 是 RSS 2.0 中 包含 的 元 素 ， 其 中 , 在 生成 RSS 文件 时 不 会 应 用 到 所 有 的 
元 素 ， 而 是 根据 实际 的 需求 应 用 不 同 的 元 素 。 
在 RSS 阅读 器 中 读 取 的 RSS 文件 中 包含 的 元 素 如 下 : 
Iss 根 元 素 。 
channel (频道 ) 元 素 。 该 元 素 包含 如 下 3 个 子 元 素 。 
title: 网 站 新 闻 频 道 的 标题 。 
link: 此 频道 相关 的 HIML 超 链 接 。 
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description: 频道 内 容 的 摘要 。 
item 项目) 元 素 。 该 元 素 包含 如 下 4 个 子 元 素 。 
title: 新 闻 标签 。 
link: 阅读 新 闻 正文 的 超 链接 。 
description: 新 闻 内 容 的 摘要 。 


pubDate: 新 闻 的 发 布 日 期 。 


问题 225 ”如何 创建 支持 RSS 阅读 的 站 点 ? 


问题 前 述 
因为 PHP 是 面向 Web 开发 的 ， 而 RSS 阅读 器 这 个 模块 似乎 更 适合 通过 应 用 程序 来 完成 ， | 
所 以 作为 PHP 程序 员 更 应 该 了 解 的 是 如 何 让 网 站 支持 RSS 阅读 ， 而 不 是 如 何 更 好 地 去 创建 开 | 
发 RSS 阅读 器 。 还 有 一 个 更 直接 的 原因 就 是 ， 如 果 想 使 用 RSS 阅读 器 ， 似 乎 从 网 上 下 载 更 方 | 
便 ， 更 快捷 。 | 
专家 解答 
要 让 站 点 支持 RSS 阅读 ， 必 须 考虑 以 下 几 个 问题 。 
(1) 将 什么 样 的 数据 作为 RSS 阅读 的 内 容 。 | 
(2) RSS 阅读 器 读 取 的 是 什么 格式 的 文件 。 
(3) 如 何 将 数据 库 中 的 数据 写 入 文件 中 。 
(4) 如 何 才能 保证 RSS 阅读 器 中 读 取 的 是 最 新 的 数据 。 
创建 支持 RSS 阅读 的 站 点 
带 着 上 述 的 疑问 开始 创建 一 个 能 够 支持 RSS 阅读 的 程序 , 这 里 创建 的 是 一 个 用 于 统计 编程 | 
技巧 的 程序 。 通 过 该 程 序 将 每 个 人 在 编程 中 总 结 出 的 经 验 技巧 进行 统计 ， 并 存储 到 数据 库 中 ， | 
然后 再 输出 这 些 技巧 让 大 家 共享 资源 。 其 运行 结果 如 图 11.2 所 示 。 


[5 


如何 通过 Pi 也 娃 am] 文 件 (明日 科 法 ) 
得 法 与 标点 符号 的 使 用 (质量 部 } 


WR:5 /6 页 记录 :加 条 3F: 首页 1 23458 Bd 


rp | 


| 

图 11.2 编程 技巧 整合 的 运行 结果 | 
有 关 该 程序 的 创建 操作 ， 这 里 将 不 作 讲解 ， 完 整 代码 可 参考 配套 资源 中 的 内 容 。 这 里 将 编 | 

程 技巧 的 标题 和 内 容 作 为 RSS 阅读 的 内 容 , 所 以 接 下 来 要 做 的 就 是 如 何 将 编程 技巧 的 标题 和 内 | 
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Fw 
2 
容 添加 到 RSS 文件 中 ， 并 且 保 证 RSS 文件 中 的 数据 是 最 新 的 。 

首先 ， 保 证 RSS 文件 中 的 内 容 是 最 新 的 ， 方 法 为 在 数据 被 添加 后 直接 更 新 RSS 文件 。 
| 在 index_okphp 中 执行 向 数据 库 中 添加 编程 技巧 的 操作 ， 并 调用 rssphp 文件 重新 生成 一 
会 和 | 个 标准 格式 的 XML 文件， 同时 更 新 XML 文件 中 的 数据 。index okphp 的 关键 代码 如 下 : 


| <2php 
We header("Content-Type:text/html: charset=utf-8"); /设置 编码 格式 


include_once("conn/conn php"): // 包 含 数据 库 连 接 文件 
S$dates=date("Y-m-d"); // 设 置 时 间 
这 $_POST[submit] 一 "提交 "){ // 浏 断 是 否 执行 提交 操作 


| $query="insert into tb_rss_database (tb Tss_subjecttb_ rss_contenttb_rss_authortb rss_department,， 
| 也 Tss date) values(™.$_POST[zhuti].",".$_POST['neirong].",".$ POST[user]""-$ POST[bumen]",$dates) 


/定义 添加 语句 
$result=mysql_query($query,$conn); /执行 添加 操作 
if($result){ 
include_once("rss.php"); // 包 仿生 成 RSS 的 文件 ， 重 新 生成 RSS 文件 
echo "<script> alert (文件 上 传 成 功 ); window.location.href='index.php';</script>"; 
jelse{ 
echo "<script> alert (文件 上 传 失败 0); window.location.href='index.php';</script>"; 
} 
1 
> 


然后 ， 将 数据 添加 到 RSS 文件 中 ， 并 且 生成 一 个 标准 格式 的 RSS 2.0 文件 ， 文 件 的 后 级 
是 .xml。 创 建 XML 文件 的 操作 是 在 rss.php 文件 中 进行 的 ， 其 操作 步骤 如 下 : 
(1) 通过 服务 器 变量 获取 头 信息 的 内 容 和 当前 页 面前 一 页 的 URL 地 址 以 及 当前 的 时 间 。 


代码 如 下 : 
$server_ ip=$_SERVER['SERVER_ NAME']: // 获 取 服 务 器 下 地址 
$request_url=str_ireplace("index_ok.php","look_content.php",$_SERVER['REQUEST_URI"); 
// 获 取 实 例 的 相对 路 径 
Shttp="http://".$server_ip.$request_url; // 定 义 一 个 访问 路 径 


S$date_time=date("Y-m-d H:i:s"):; 


(2) 通过 new DomDocument 创建 DOM 对 象 ， 创 建 XML 文件 的 根 节点 ， 并 且 为 根 节点 
添加 属性 及 属性 值 。 代 码 如 下 : 


| $dom = new DomDocument('1.0',"utf-8"): // 创 建 DOM 对 象 
$object = $dom->createElement('rss"): // 创 建 根 节点 rss 
| $dom->appendChild($object): /将 创建 的 根 节点 添加 到 DOM 对 和 象 中 
| $typel = $dom->createAttribute('xmilns:rdf ): // 创 建 一 个 节点 属性 xmlns:rdf 
| $object->appendChild($typel); /将 属性 追加 到 rss 根 节点 中 
| $typel_value = $dom->createTextNode('http://www.w3.0org/1999/02/22-rdf-syntax-ns#'"):; 
| /创建 一 个 属性 值 
| $typel->appendChild($typel_value); // 将 属性 值 赋 给 属性 xmlns:rdf 
| $type2 = $dom->createAttribute(xmlns:dc); /创建 一 个 节点 属性 xmlns:dc 
| Sobject->appendChild(Stype?); // 将 属性 追加 到 rss 根 节点 中 
| $type2_value = $dom->createTextNode(http://purlorg/dc/elements/1.1/); 
| // 创 建 一 个 属性 值 
| $type2->appendChild($type2_value); // 将 属性 值 赋 给 属性 xmlns:dc 
| $type3 = $dom->createAttribute(xmlns:taxo): // 创 建 一 个 节点 属性 xmlns:taxo 
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S$object->appendChild($type3): /将 属性 追加 到 rss 根 节点 中 

$type3_value = $dom->createTextNode('http://purl.org/rss/1.0/modules/taxonomy/"); 
// 创 建 一 个 属性 值 

Stype3->appendChild($type3_value); // 将 属性 值 赋 给 属性 xmlns:taxo 

$type4 = $dom->createAttribute(version ): // 创 建 一 个 节点 属性 version 

$object->appendChild($type4): /将 属性 追加 到 rss 根 节点 中 
$type4_value = $dom->createTextNode('2.0); ” // 创 建 一 个 属性 值 
$type4->appendChild($type4_value): // 将 属性 值 赋 给 属性 version 


(3) 创建 channel 节点 ， 这 是 RSS 2.0 中 的 channel 元 素 及 其 子 元 素 。 其 中 包括 title link、| 
description 和 dc:creator 4 个 子 元 素 。 代 码 如 下 : | 


$channel = $dom->createElement('channel)): /创建 节点 channel | 
S$object->appendChild($channel); /将 节点 channel 追加 到 rss 根 节点 下 | 
Stitle = $dom->createElement('title'): // 创 建 节点 title | 
$channel->appendChild($title): /将 节点 追加 到 channel 节点 下 
$title_value = $dom->createTextNode(' 明 日 科技 );/ 创 建 元 素 值 
Stitle->appendChild($title_value); // 将 值 赋 给 title 节点 
S$link = $dom->createElement(link'); // 创 建 节点 link 
$channel->appendChild($link):; /将 节点 追加 到 channel 节点 下 | 
S$link_value = $dom->createTextNode('http://www.mingrisoft.com'); // 创 建 元 素 值 | 
$link->appendChild($link value): // 将 值 赋 给 link 节点 | 
$description = $dom->createElement('description); 。 // 创 建 节点 description 
$channel->appendChild($description); /将 节点 追加 到 channel 节点 下 


$description_value = $dom->createTextNode(' 明 日 科技 ): ”// 创 建 元 素 值 
$description->appendChild($description_value); // 将 值 赋 给 description 节点 
$dc_creator = $dom->createElement('dc:creator’); // 创 建 节点 dc:creator 
$channel->appendChild($dc_creator): /将 节点 追加 到 channel 节点 中 
$dc _creator_ value = $dom->createTextNode('http://www.mingrisoft.com'");// 创 建 元 素 值 
$dc_creator->appendChild($dc_creator_value); /将 值 赋 给 dc:creator 节点 


(4) 创建 item 节点 。 从 数据 库 中 读 取出 编程 技巧 的 数据 ， 并 且 应 用 while 循环 输出 数据 

库 中 的 数据 ， 将 其 作为 item 节点 中 子 节点 的 元 素 值 。 这 里 创建 的 是 RSS 2.0 中 的 item 元 素 及 | 

其 子 元 素 ， 其 中 包括 title、link、description 和 pubDate 4 个 子 元 素 。 代 码 如 下 : | 
$sql=mysql_query("select * from tb_rss_database order by tb_rss_id desc"); /从 数据 库 中 读 取 数 据 


while($myrow=mysql fetch array($sql){ /循环 输出 数据 库 中 数据 | 
S$item = $dom->createElement('item'): // 创 建 节点 item | 
Schannel->appendChild(S$item); /将 item 追加 到 channel 节点 下 | 
Sitem title = $dom->createElement('title’): // 创 建 title 节点 | 
S$item->appendChild(Sitem title): /将 节点 追加 到 item 节点 下 | 
$item link = $dom->createElement(link'); // 创 建 link 节点 | 
S$item->appendChild($item link): /将 节点 追加 到 item 节点 下 | 
$item_description = $dom->createElement('description");// 创 建 description 节点 | 
S$item->appendChild($item_description): /将 节点 追加 到 item 节点 中 | 
$item_pubDate = $dom->createElement(pubDate); /创建 节点 pubDate | 
Sitem->appendChild($item pubDate): /将 节点 追加 到 item 节点 下 | 


$title_value = $dom->createTextNode($myrow['tb_rss_subject]); ” // 创 建 元 素 值 | 

Sitem title->appendChild($title_value): // 将 值 赋 给 title 节点 

$link value = $dom->createTextNode($http."?Imbs=".$myrow['tb_rss_1d"]); 
// 创 建 元 素 值 

Sitem link->appendChild($link_value):; // 将 值 赋 给 link 节点 
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S$description=mb substr($myrow['tb_rss_content"],0,80,"utf-8"); 
/截取 该 字段 中 的 前 80 个 字符 
$description value = $dom->createTextNode($description): /创建 元 素 值 
$item_ description->appendChild($description value); // 将 值 赋 给 description 节点 
S$pubDate value = $dom->createTextNode($date time); // 创 建 元 素 值 
$item pubDate->appendChild($pubDate_value); // 将 值 赋 给 pubDate 节点 
} 


(5) 通过 $dom->saveXML (方法 生成 XML 文档 ， 并 应 用 file_put_contents0 函 数 将 生成 的 
XML 文档 保存 到 Rss.xml 文件 中 。 代 码 如 下 : 


$modi = $dom->saveXML(); /生成 XML 文档 
file_ put_contents('Rss.xml',$modi); 证 将 对 象 保存 到 Rss.xml 文档 中 */ 


至 此 , 一 个 支持 RSS 阅读 的 站 点 创建 完毕 ,只 要 在 站 点 中 添加 新 的 数据 ， 存 储 在 根 目录 下 
的 XML 文件 就 会 被 更 新 一 次 ， 从 而 保证 了 RSS 阅读 器 中 能 读 取 到 站 点 中 最 新 的 数据 。 


问题 226 ”如 何 动态 创 建 XML 文件 ? 


问题 前 述 
通过 XML 可 以 将 数据 存储 在 一 个 文档 中 ， 并 且 可 以 实现 数据 格式 的 转换 和 传递 。 那 么 如 
何在 PHP 中 应 用 XML 文档 ? 如 何 通 过 PHP 动态 创建 XML 呢 ? 
专家 解答 
在 PHP 中 动态 创建 XML 文档 的 方法 有 很 多 ,下面 讲解 如 何 通过 DOM 对 象 动态 创建 XML 
文件 。 通 过 DOM 类 库 创建 XML 文档 应 用 到 的 方法 如 下 : 
(1) 通过 new DomDocumentO 实 例 化 一 个 DomDocument 对 象 。 语 法 如 下 : 


$dom = new DomDocument('1.0','gb2312"); /创建 DOM 对 象 
(2) 通过 $dom->createElement 创建 一 个 新 元 素 节 点 。 语 法 如 下 : 


class DomDocument { 
DomElement createElement ( string name [, string value] ) 
} 


参数 name 是 创建 的 元 素 的 名 称 ， 参 数 value 是 创建 元 素 的 值 。 如 果 成 功 则 返回 一 个 元 素 
节点 对 象 ， 和 否则 返回 一 个 错误 。 
(3) 通过 $dom->appendChild 实现 元 素 节点 的 附加 。 语 法 如 下 : 
class DomNode { 
DomNode appendChild ( DOMNode newnode ) 
} 
参数 newnode 是 将 要 附加 的 节点 ， 返 回 更 多 的 节点 。 
(4) 通过 $dom->createAttribute 创建 一 个 新 的 属性 。 语 法 如 下 : 
class DomDocument { 


DomAttr createAttribute ( string name ) 
} 
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参数 name 是 属性 的 名 称 。 如 果 成 功 则 返回 
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-个 属性 对 象 ， 否 则 返回 - 


(5) 通过 $dom->createTextNode 创建 一 个 属性 值 。 语 法 如 下 : 


class DomDocument { 
DomText createTextNode ( string content ) 
} 


参数 content 是 属性 的 值 。 如 果 成 功 则 返回 


一 个 属性 值 对 象 ， 否 则 返回 一 个 错误 。 


(6) 通过 $dom->saveXML 生成 一 个 XML 文档 。 语 法 如 下 : 


class DomDocument { 
string saveXML ( [DOMNode node [, int options]] ) 
' 


参数 node 指定 生成 的 节点 变量 名 称 ; 参数 options 是 附加 的 值 。 如 果 成 功 则 返 


文档 ， 否 则 返回 一 个 错误 。 
还 6 个 方法 是 在 创建 XML 文档 中 需要 使 用 的 。 
应 用 示例 


回 一 个 XML 


下 面 应 用 上 述 的 方法 创建 一 个 XML 文档 。 其 关键 代码 如 下 : 


<2php 
header('Content-Type:text/html:charset=utf-8"):; 
$dom = new DomDocument('1.0',"utf-8"); 
S$object = $dom->createElement('‘object'):; 
$dom->appendChild($object); 

$book = $dom->createElement('book’):; 
S$object->appendChild($book); 


$computerbook = $dom->createElement('computerbook’); 


$book->appendChild($computerbook): 

$type = $dom->createAttribute('type’); 
$computerbook->appendChild($type); 
$type_value = $dom->createTextNode(computer'); 
$type->appendChild($type_value):; 

$bookname = $dom->createElement(‘bookname’): 
$computerbook->appendChild($bookname); 


$bookname_value = $dom->createTextNode(' 学 通 PHP24 堂 课 ); 


$bookname->appendChild($bookname_value); 
$xml file=$dom->saveXML(; 


echo $xml file; 
file_put_contents('01.xml',$xml file); 
?> 
运行 结果 如 图 11.3 所 示 。 
专家 点 评 


DOM (Document Object Model)， 文 档 对 象 模 型 ， 
通过 树 型 结构 模式 来 遍历 XML 文档 。 使 用 DOM 遍历 
文档 的 好 处 是 不 需要 标记 就 可 以 显示 全 部 内 容 ， 但 缺 
点 同样 明显 ， 就 是 非常 消耗 内 存 。 
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/设置 编码 格式 

/创建 DOM 对 象 

/创建 根 节点 object 

/将 创建 的 根 节 点 添加 到 dom 对 象 中 
// 创 建 节点 book 

/将 节点 book 追加 到 dom 对 象 中 
/创建 节点 computerbook 

/将 computerbook 追加 到 dom 对 象 中 
/创建 一 个 节点 属性 fype 

/将 属性 追加 到 computerbook 元 素 后 
/创建 一 个 属性 值 

// 将 属性 值 赋 给 type 

// 创 建 节点 bookname 

/将 节点 追加 到 dom 对 和 象 中 
// 创 建 元 素 值 

// 将 值 赋 给 节点 bookname 


// 输 出 XML 文件 
// 将 XML 文档 保存 到 指定 的 文件 中 


人 attp: //192. 168. 1.59/mr/1S/ rb/ TOP SoD Te 


ckname> 


EE 


图 11.3 动态 创建 XML 文档 


人 Pp 六 须知 道 的 300 个 问题 
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问题 227 ”如 何 动态 读 取 XML 文件 ? 
问题 阐述 


既然 通过 XML 可 以 将 数据 存储 在 一 个 文档 中 ， 那 么 如 何 读 取 这 个 文档 中 的 数据 呢 ? 
专家 解答 
可 以 应 用 DOM 类 库 中 的 getElementsByTagName() 方 法 动态 读 取 文档 中 的 数据 。 


| DOMDocument->getElementsByTagName() 方 法 用 于 获取 标签 的 名 称 ， 返 回 值 为 一 个 新 的 


DOMNodeList 对 象 。 语 法 如 下 : 
class DOMDocument { 
DOMNodeList getElementsByTagName ( string name ) 
} 
参数 name 是 XML 文档 中 节点 的 名 称 。 
应 用 示例 
本 示例 应 用 DOM 类 库 中 的 getElementsByTagName() 方 法 读 取 XML 文档 ， 并 以 for 和 
foreach 语句 作为 辅助 ， 循 环 输出 XML 文档 中 的 数据 。 本 示例 的 所 有 功能 都 是 在 index.php 文 
件 中 实现 的 ， 具 体 步 骤 如 下 : 
(1) 编写 PHP 脚本 语句 ， 创 建 一 个 Read XMLO 类 ， 继 承 DOMDocument 类 库 。 在 类 文 
件 中 定义 了 两 个 函数 : 一 个 是 构造 函数 ， 通 过 load() 方 法 读 取 存 储 在 根 目 录 下 的 Rss.xml 文件 ; 
另 一 个 是 show_xml0 函 数 ， 应 用 getElementsByTagName() 方 法 读 取 XML 文档 中 的 数据 ， 并 通 
过 for 和 foreach 语句 循环 输出 数据 。 其 关键 代码 如 下 : 


<?php 
class Read XML extends DOMDocument{ // 创 建 一 个 类 
private $root; 
public function constructO{ // 构 造 函数 
parent ::_ construct(); 
S$this->load("Rss.xml"); // 读 取 指定 的 XML 文件 
public function show_xmlO{ /| 输出 XML 文件 中 的 内 容 
$root=$this->documentElement: // 定 义 根 目录 
$node_record=$this->getElementsByTagName("item"); /获取 节点 item 下 的 标签 
$node record length=$node record->length: // 获 取 标 签 的 数量 
for($i=0:$i<$node record->length:$i++){ /根据 标签 的 数量 执行 for 循环 
$k=0; 
foreach($node record->item($i)->childNodes as $data){ // 通 过 foreach 语句 输出 标签 中 的 数据 
Sfield[$k]= $data->textContent; 
Sk+t; 
echo '<table width="672" border="1" cellpaddine="1" ce ing="1" bordercolor="#FFFFFF" 
bgcolor="#999999"><tr>"; 


echo '<td width="102" height="22" align—"right" bgcolor="#FFFFFF"> 标 题 ， </td>': 
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echo '<td width="368" bgcolor="#FFFFFF">&nbsp;'". $field[0].'</td>"; 
echo '<td width="184" align="center" bgcolor="#FFFFFF"> 时 间 : '.$field[3].'</td>'; 
echo ‘</tr><tr>"; 
echo '<td colspan="3" height="40"; style="padding-left:S5px; padding-right:5px; padding-top:5px; 
padding-bottom:Spx:" bgcolor="#FFFFFF">". $field[2].'</td>"; 
echo ‘</tr><tr>"; 
echo '<td height="25" align="right" bgcolor="#FFFFFF">URL:</td>"; 
echo '<td colspan="2" bgcolor="#FFFFFF">&nbsp;&nbsp;'. $field[1].'</td>"; 


echo '</tr></table>": 
echo '</table>'; 
} 
1 
} 
> 


(2) 创建 table 表格 , 添加 查看 XML 文件 的 超 链接 。 实 例 化 Read_ XML 类 , 并 执行 show_ 
xml0 方 法 ， 完 成 XML 文件 的 读 取 。 其 关键 代码 如 下 : 


<table width="672" height="30" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" 
bgcolor="#999999"> 
<tr> 
<td height="30" align="center" bgcolor="#FFFFFF">PHP 读 取 XML 文件 &nbsp:&nbsp:&nbsp; 
&nbspi&nbsp:&nbspi&nbsp:&nbsp:<a href="Rss.xml" class="STYLE3"> 查 看 XML 文件 内 容 </a></td> 
</tr> 
<?php 
S$Read XMl=new Read_ XML; // 类 的 实例 化 
S$Read XMI->show xml(); // 执 行 类 中 方法 
J 
</table> 


(3) 在 示例 的 根 目录 中 创建 一 个 Rss.xml 文件 ， 并 将 其 作为 读 取 的 内 容 。 
通过 PHP 动态 读 取 XML 文档 中 的 数据 ， 其 运行 结果 如 图 11.4 所 示 。 


PEP 该 取 XJL 文 件 查看 XL 文件 内 容 


标题 | 《Asp. net 编 程 词典 》 服 务 方式 时 间 ，2009-10-16 10:18:01 


方式 一 : 电话 400-675-1066 方式 二 ，TQ: .net 编 程 词典 
方式 三 ，.netQQ ，200958603 方式 四 : 论坛 bbs. arbccd. con 
TRL:| http://www mrbccd con 


标题 | 《Asp. net 编 程 词典 》 服 务 时 间 时 间 ，2009-10-16 10:18:01 
周一 至 周 五 早 8 点 到 晚 24 点 


URL:| http://ww. arbecd. con 


<item> 
<title> 《Asp.net 编 程 词典 》 服 务 方式 </title> 
<link>http:/ /www.mrbccd.com</Iink> 
<description> 方 式 一 : 电话 : 400-675-1066 方式 二 TQ: .net 编程 词 典 
方式 三 : .netQQ : 200958603 方式 四 : 论坛 
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bbs.mrbccd.com</description> 
<pubDate>2009-10-16 10:18:01</pubDate> 
</item> 


<link>http:/ /www.mrbccd.com</INk> 
<description> 周 一 至 周 五 早 8 点 到 腊 24 点 </description> 
<pubDate>2009-10-16 10:18:01</pubDate> 

</item> 


citem> 
| <title> 《Asp net 编 程 词典 》 服 务 时 间 -/title> 


图 11.4 PHP 读 取 XML 文件 
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到 人 ip 六 预 知道 的 300 个 问题 


问题 228 ”如 何 查询 XML 文件 中 的 数据 ? 


入】 “问题 间 述 


如 何 查询 XML 文件 中 的 数据 ? 
| 专家 解答 


| 在 PHP 中 实现 对 XML 的 查询 操作 ， 应 用 的 是 DOM 类 库 中 的 DOMXPath->query() 方 法 。 
| 然后 通过 for 和 foreach 语句 循环 输出 查询 结果 。 
| (1) DOMXPath->query() 方 法 用 于 查询 出 XML 文件 中 符合 条 件 的 特定 节点 ， 返 回 值 为 一 
| 个 节点 列表 对 象 。 语 法 如 下 : 
| class DOMXPath { 
DOMNodeList query ( string expression [, DOMNode contextnode] ) 

} 
| 其 参数 expression 是 执行 的 查询 语句 。 若 成 功 则 返回 一 个 节点 列表 对 象 DOMNodeList; 若 
| 失败 ， 则 返回 一 个 空 的 节点 列表 对 象 DOMNodeList。 
| 本 示例 中 使 用 的 查询 语句 如 下 ; 

Witemftitle=$search title]" 


查询 出 item 节点 中 ，title 子 节点 的 值 等 于 变量 $search title 的 特定 节点 。 
| (2) $node_Record 对象， 查询 返回 的 节点 列表 对 象 ， 通 过 for 和 foreach 语句 循环 输出 
| $node_ Record 对 象 中 的 数据 。 
| DOMDocument->getElementsByTagName() 方 法 用 于 获取 标签 的 名 称 ， 返 回 值 为 一 个 新 的 
| DOMNodeList 对 象 。 语 法 如 下 : 
| class DOMDocument { 
DOMNodeList getElementsByTagName ( string name ) 


b: 
参数 name 是 XML 文档 中 节点 的 名 称 。 
应 用 示例 


| 通过 DOM 类 库 中 的 DOMXPath->query() 方 法 查询 XML 文档 中 的 数据 ， 并 且 通 过 for 和 
| foreach 语句 循环 输出 查询 结果 。 其 具体 步骤 如 下 : 

| (1) 通过 jQuery 技术 动态 输出 整个 XML 文档 中 的 数据 。 应 用 存储 在 js 文件 夹 下 的 funjs 
| 脚本 文件 和 jquery-1.3.2js 框架 包 。 

| (2) 创建 mdex.php 文件 。 首 先 ， 创 建 Search XML 类， 继承 DOM 类 库 ， 定 义 构 造 方法 
| 读 取 指定 的 XML 文件 ; 定义 Select XML (方法 ， 通 过 DOMXPath->query0 方 法 完成 对 XML 
| 的 查询 操作 。 其 关键 代码 如 下 : 


<?php 
class Search XML extends DomDocument{ /创建 类 ， 继 承 DOM 类 库 
public function constructO { /构造 方法 ， 打 开 指定 的 XML 文件 


parent:: _constructO: 
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S$this->load("Rss.xml"): ! 

} | 
public function Select XMIL($search title){ ”// 定 义 方法 , 执行 查询 操作 , 参数 查询 的 关键 字 | 
$xpath = new DOMXPath(S$this): /应 用 DOMXPath 方法 进行 查询 | 
$node Record = $xpath->query("//item[title='$search title]"); /执行 查询 | 
for($i=0:$i<$node_ Record->length:$i++){ // 循 环 输出 查询 结果 | 

Se=0; | 


foreach($node Record->item($i)->childNodes as $node){ ”// 读 取 数 组 中 的 数据 
Sfield[$g]=$node->textContent:; 
Sgtt; 


} 
echo '<table width="672" border="1" cellpadding="1" cellspacing="1" bordercolor= 
"#FFFFFF" bgcolor="#999999"><tr>"; 
echo '<td width="102" height="22" align="right" bgcolor="#FFFFFF"> 标 题 : </td>'; 
echo '<td width="368" bgcolor="#FFFFFF">&nbsp;". $field[0].'</td>"; 
echo '<td width="184" align="center" bgcolor="#EFFFFF"> 时 间 : ".$field[3].'</td>"; 
echo '</tr><tr>': 
echo '<td colspan="3" height="40" style="padding-left:Spx: padding-right:Spx; 
padding-top:Spx; padding-bottom:5px;" bgcolor="#FFFFFF">&nbsp;&nbsp;'". $field[2].'</td>"; 
echo '</tr><tr>'; 
echo '<td height="25" align="right" bgcolor="#FFFFFF">URL:</td>"; 
echo '<td colspan="2" bgcolor="#FFFFFF">&nbsp;&nbsp;". $field[1].'</td>"; 
echo '</tr></table>"; 
echo '</table>"; 


} 


?> 


然后 ,通过 Script 脚本 标记 调用 funjs 和 jquery-1.3.2js 脚本 文件 ,完成 XML 文件 的 读 取 。 
其 关键 代码 如 下 : 


‘<script src="js/jquery-1.3.2.js"></script> 
<script src="js/fun.js"></script> 


接着 ,创建 form 表单 ， 添 加 表单 元 素 ， 完 成 查询 关键 字 的 提交 。 其 关键 代码 如 下 : 


<form name="forml" method="post" search_title="index.php"> 
<tr> 
<td height=35 align="center" bgcolor="#CCCCCC"><span class="STYLE select5"> 请 输入 标 
题名 称 : </span> 
<input name="search title" type="text" id="search title"> 
<input type="submit" name="Submit" value=" 提 交 "> 
<a hre 全 "Rss.xml" class="STYLE3"> 查 看 XML 文件 内 容 </a> 
</td> 
</tr> 
</form> 


最 后 ， 实 例 化 Search_ XML 类， 执行 Select_ XML() 方 法 ， 将 表单 中 提交 的 关键 字 作 为 参数 
传递 到 Select XML( 方 法 中 ， 完 成 对 XML 的 查询 。 其 关键 代码 如 下 : 


<?php 
ifisset($_ POST['Submit]) && isset($_POST['search title]) && $ POST[search title]!=""){ 
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| // 判 断 查 询 提 交 按 钮 是 否 为 真 
| $Search XML = new Search XML /实例 化 类 
| $Search XML->Select XML($_ POST[search title]): /执行 类 中 方法 ， 并 对 提交 的 
| 关键 字 进 行 编码 转换 
| jelse{ // 如 果 提 交 按 钮 和 查询 关键 字 为 空 ， 则 输出 如 下 内 容 
| D3 
| <div> 
p>enbsp:<p> 
| </div> 
| <2php 
| ?> 


(3) 在 示例 的 根 目录 中 创建 一 个 Rss.xml 文件 ， 作 为 查询 的 内 容 。 
运行 结果 如 图 11.5 所 示 。 


i 入 生 是 多 所 环 生机 人 月 。。 提交 | 查看 ML 文人 内容 


标题 :| 和 法 与 标点 符号 的 使 用 时 间 : 2010.41-27 0:5639 
语法 第 十 七 条 : 图 书 中 常见 的 语法 将 铺 ， 大致 可 以 分 为 : 词性 涡 用 ,数量 表达 混乱 ， 指 


UPL| ittpyl92.163139knrl SenhMOl_diytss phpylsbs=5 


图 11.5 PHP 查询 XML 文件 
问题 229 ”如 何 动态 更 新 XML 中 的 数据 ? 


问题 讲述 
如 何 动态 更 新 XML 中 的 数据 ? 
专家 解答 


要 想 实 现 动 态 更 新 XML 中 的 数据 ， 主 要 应 用 DomDocument 类 中 的 createTextNode() 和 
replaceChild() 方 法 完成 对 子 节点 值 的 修改 。 


(1) createTextNode() 方 法 用 于 创建 一 个 新 的 文本 节点 。 语 法 如 下 : 
| class DOMDocument { 
| DOMText createTextNode ( string content ) 
| } 
| 参数 content 是 文本 的 内 容 。 若 成 功 ， 则 返回 一 个 文本 节点 ; 若 失败 ， 则 返回 一 个 错误 信息 。 
说明， 


| TextNode() 方 法 创建 的 文本 节点 不 会 出 现在 文档 中 ， 除 非 使 用 DOMNode->appendChild() 
方法 将 其 添加 到 文档 中 。 


(2) replaceChild0) 方 法 用 于 修改 一 个 子 节点 。 语 法 如 下 : 


class DOMNode { 
DOMNode replaceChild ( DOMNode newnode, DOMNode oldnode ) 
中 
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该 方法 实现 将 当前 的 oldnode 节点 蔡 换 为 newnode 节点 。 

应 用 示例 | 

在 PHP 中 修改 XML 文件 , 需要 综合 地 运用 读 取 和 查询 XML 文件 的 技术 。 首先 读 取 XML | 
文件 中 的 数据 ， 然 后 根据 超 链接 中 传递 的 标题 ， 通 过 查询 定位 到 指定 的 XML 文件 ， 最 终 实现 | 
对 XML 文件 的 修改 。 其 具体 步骤 如 下 : | 
(1) 完成 类 的 创建 和 方法 的 定义 。 创 建 Amend_ XML 类 ,继承 DOM 类 库 。 其 中 定义 了 4 

个 方法 。 分 别 是 构造 函数 ， 完 成 对 指定 XML 文件 的 读 取 ; show_amend() 方 法 读 取 并 输出 XML | 
文件 中 的 数据 ，search_xml() 方 法 执行 查询 操作 ， 并 将 查询 结果 输出 到 form 表单 中 ， 作 为 要 修 | 
改 的 内 容 ，amend xml() 方 法 完成 对 指定 节点 元 素 的 修改 操作 。 其 关键 代码 如 下 : | 


<?php | 
class Amend XML extends DomDocument{ ”// 定 义 类 文件 ， 继 承 DOM 类 对 象 | 
public function __ constructO { /构造 函数 | 
parent:: _constructO: 
Sthis->load("Rss.xml"); // 读 取 存 储 留 言 信息 的 XML 文档 Rss.xml 
} 
/省 略 了 部 分 代码 
public function amend xml($title.$link.$description){ /定义 保存 数据 的 方法 
$xpath = new DOMXPath($this): // 查 询 待 修改 的 记录 


$Node_Record = $xpath->query("//item[title=$title]");// 执 行 查询 操作 ， 同 样 以 标题 为 依据 

S$Replace[0]=Stitle; 

S$Replace[1]=$link: 

$Replace[2]=$description; 

$d=0; | 

foreach ($Node_Record->item(0)->childNodes as $articles) { | 
$Node_newText = $this->createTextNode($Replace[$d]); // 对 提交 的 修改 数据 进行 转 码 | 
$articles->replaceChild($Node_newText,$articles->lastChild);// 执 行 修 改 操作 | 
$d++; 


} 

echo "<script>alert(' 修 改 成 功 ');location.hre 仁 ".$_SERVER['PHP_SELF']."</script>"; 

Sthis->save("Rss.xml"):; | 
} | 
} | 


?> 


(2) 实例 化 Amend_ XML 类 , 应 用 switch 语句 ， 根 据 参 数 传递 的 不 同 值 执行 不 同 的 方法 ， 
完成 对 XML 文件 的 各 种 操作 。 其 关键 代码 如 下 : | 


<?php | 
$amend xml = new Amend XML /实例 化 类 | 
$Action =""; // 定 义 变 量 | 
iftisset($_GET['Action'])) // 判 断 超 链接 传递 的 值 | 
$Action =$_GET['Action']: // 获 取 超 链 接 传递 的 值 | 
switch($Action){ // 根 据 超 链接 传递 的 值 进行 判断 | 
case "show_amend": /浏览 | 

$amend xml->show_amend(): 
break: | 
case "search xml": // 查 询 | 
$amend xml->search xml($ GET['title]): | 
break: | 
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case "amend xml": // 修 改 
soma xml->amend xml($ GET['title'],$ POST[link].$ POST[description]); 


全 六 Ce xml->show_amend(); 


” ? 
| | 运行 结果 如 图 11.6 所 示 。 


标题 | 和 后 通 过 Pap 人 下 ont 文件 E 
谤 接地 址 。 [Tp es i 
[Ea 请 参考 吉林 省 明日 科 技 有 限 公 EIT 书 。 司 
目 
修改 数据 
图 11.6 PHP 修改 XML 文件 
问题 230 ”如何 动态 删除 XML 中 的 数据 ? 
| 问题 章 述 
| 如 何 动态 删除 XML 中 的 数据 ? 
| 专家 解答 
| 应 用 DOM 类 库 中 的 removeChild0 方 法 删除 XML 文档 中 指定 的 子 节点 。removeChild() 方 
”法 的 语法 如 下 : 
| class DOMNode { 
DOMNode removeChild ( DOMNode oldnode ) 
} 
| 参数 oldnode 为 被 删除 的 子 节点 。 若 成 功 ， 则 返回 被 删除 的 节点 。 删 除 成 功 后 ， 需 要 通过 
| save() 方 法 重新 保存 文件 。 
| 应 用 示例 


| 在 PHP 中 动态 删除 XML 文件 中 的 数据 主要 应 用 到 3 个 自 定义 的 方法 : 一 是 构造 函数 ， 完 
| 成 读 取 指定 XML 文件 的 操作 ; 二 是 delete_ xml0 方 法 , 执行 XML 的 删除 操作 ; 三 是 show_xml0 
| 方法 ， 读 取出 XML 文档 中 的 数据 。 其 具体 步骤 如 下 : 

| (1) 在 indexphp 文件 中 ， 创 建 Delete XML 类 ， 继 承 DOM 类 库 ， 定 义 构造 方法 完成 对 
| XML 的 读 取 ， 定 义 delete_ xml0 方 法 执行 删除 XML 节点 的 操作 ;定义 show_xml() 方 法 实现 输 
| 出 XML 文档 中 的 数据 。 其 关键 代码 如 下 : 


<?php 

class Delete XML extends DomDocument{ ”// 定 义 类 文件 ， 继 承 DOM 
private $Root:; // 定 义 成 员 变 量 
public function constructO { // 定 义 构 造 方法 


| parent:: _constructO: 
J 356 


$this->load("Rss xml"): // 读 取 XML 文档 Rss.xml 


} 

public function delete xml($title){ /定义 删除 XML 节点 的 方法 ， 以 超 链接 传递 的 值 为 参数 
$Root=$this->documentElement: // 定 义 根 目录 
$xpath = new DOMXPath($this): // 应 用 XPath 方法 查询 数据 
$Node_Record= $xpath->query("//item[title='$title]"); // 以 标题 为 条 件 进行 查询 
$Root->removeChild($Node Record->item(0)); // 根 据 查询 结果 ， 执 行 删除 操作 
Sthis->save("Rss.xml"); // 重 新 保存 文件 
echo "<script>alert( 删 除 成 功 "):location.href=".$_SERVER['PHP_SELF']."</script>"; 


} 
public function show_xmlO{ /定义 输出 XML 文档 的 方法 | 
$Node Record= $this->getElementsByTagName("item"); /获取 节点 | 
$Node Record Length =$Node Record->length; // 获 取 节 点 的 个 数 | 
for($i=0;$i<$Node_Record->length:$it+) { // 执 行 for 循环 
S$K=0; 


foreach ($Node Record->item($i)->childNodes as $articles) {// 应 用 foreach 语句 输出 数据 
Sfield[$K]=$articles->textContent; 
$K++; 


} 
echo '<table width="672" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" 
bgcolor="#999999"><tr>"; 
echo '<td width="102" height="22" align="right" bgcolor="#FFFFFF"> 标 题 : </td>'; 
echo '<td width="368" bgcolor="#FFFFFF">&nbsp:". $field[0]."</td>"; 
echo '<td width="184" align="center" bgcolor="#FFFFFF"><a href="?Action=delete_ 
xml&title='.$field[0]."> 删 除 </a></td>'; 
echo '</tr><tr>'; 
echo '<td colspan="3" height="40" style="padding-left:5px:; padding-rightSpx: padding-top:Spx; 
padding-bottom:5px:" bgcolor="#FFFFFF">&nbsp:;&nbsp;'. $field[2].'</td>"; 
echo '</tr><tr>"; 
echo '<td height="25" align="right" bgcolor="#FFFFFF">URL:</td>'; 
echo '<td colspan="2" bgcolor="#FFFFFF">&nbsp:é&nbsp;". $field[1].'</td>"; 
echo '</tr></table>'; 
echo '</table>"; 


} 
} 


es 


(2) 实例 化 Delete XML， 应 用 switch 语句 ， 根 据 传递 的 不 同 参数 分 别 执行 不 同 的 方法 ， 
完成 对 XML 的 删除 和 输出 操作 。 其 关键 代码 如 下 : | 


<2php | 


$Action =""; // 定 义 变量 | 
SDeleteXML = new Delete XML /实例 化 删除 类 | 
ifisset($_GET['Action]) // 判 断 超 链接 传递 的 值 | 
$Action =$_GET['Action']: /获取 超 链接 传递 的 值 | 
switch($Action){ // 根 据 超 链接 传递 的 值 进 行 判 断 | 
case "delete_ xml": /删除 | 
$DeleteXML->delete xml($_GET['itle]): /执行 删除 操作 
break:; 
default: 


$DeleteXML->show_xml0: 
*。359。 
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} 
Be 
| (3) 在 根 目录 下 创建 Rss.xml 文件 ， 定 义 被 操作 的 内 容 。 
会 内 | 运行 本 示例 ， 结 果 如 图 11.7 所 示 。 单 击 结果 页 面 中 的 “删除 ” 超 链 接 ， 即 可 删除 指定 的 
和 XML 数据 。 


标题 ，| 如 末 使 月 es 网 交 汪 


esd 
| 攻 | 嘱 | 
| 
| 如 何 饮用 Rss 周 文生 如何 亿 用 Rs 阅读 侣 如 何 使 用 Rs5 阅 该 器 
| 
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图 11.7 PHP 删除 XML 文件 
问题 231 如 何 通过 jQuery 读 取 XML 中 的 数据 ? 


问题 阐述 


| 在 提 到 读 取 XML 文档 中 的 数据 时 ， 大 家 可 能 都 会 想到 DOM 类 库 或 SimpleXML 对 象 ， 它 
| 们 都 可 以 完成 对 XML 的 读 取 操作 ， 那 么 如 何 通过 jQuery 读 取 XML 中 的 数据 呢 ? 


| 专家 解答 
| 通过 jQuery 读 取 XML 中 的 数据 应 用 到 的 技术 如 下 。 
| (1) ready(fn)， 当 DOM 载 入 就 绪 可 以 查询 及 操纵 时 绑 定 一 个 要 执行 的 函数 。 通 过 
| $(document) ready(fo) 代 蔡 body 标签 的 onload 事件 。 
| ready(fn) 
参数 和 (Function) 是 在 DOM 就 绪 时 执行 的 函数 ， 可 以 为 这 个 参数 任意 起 一 个 名 字 。 
(2) $.get0 请 求 。 语 法 如 下 : 
jQuery.get(url,[datal],[callback]) 


| GET 请 求 返回 一 个 XMLHttpRequest。 参 数 url(String) 是 待 载 入 页 面 的 URL 地 址 ; data(Map) 
| 为 可 选 参数 ， 是 待 发 送 Key/value 参数 ，callback (Function) 也 是 可 选 参 数 ， 是 载 入 成 功 时 的 回 
| 调 函数 。 
| (3) empty0 删 除 匹 配 的 元 素 集合 中 所 有 的 子 节点 ， 其 返回 值 是 jQuery。 
| (4) append(content)， 向 每 个 匹配 的 元 素 内 部 追加 内 容 。 这 个 操作 与 对 指定 的 元 素 执行 

| appendChild() 方 法 ,将 它们 添加 到 文档 中 的 情况 类 似 。 其 返回 值 为 jQuery， 参数 content (String, 

| Element jQuery) 是 要 追加 到 目标 中 的 内 容 。 

| (5) find(expr) 搜 索 所 有 与 指定 表达 式 匹 配 的 元 素 。 该 函数 是 找 出 正在 处 理 的 元 素 的 后 代 

| 元素 的 好 方法 。 其 返回 值 是 jQuery。 参数 expr (String) 是 用 于 查找 的 表达 式 。 

| (6) each(callback) 以 每 一 个 匹配 的 元 素 作为 上 下 文 来 执行 一 个 函数 ， 即 每 次 执行 传递 进 

| 来 的 函数 时 ， 函 数 中 的 this 关键 字 都 指向 一 个 不 同 的 DOM 元 素 〈 每 次 都 是 一 个 不 同 的 匹配 元 

| 素 )， 而 且 在 每 次 执行 函数 时 ， 都 会 给 函数 传递 一 个 表示 作为 执行 环境 的 元 素 在 匹配 的 元 素 集 

| 合 中 所 处 位 置 的 数字 值 作为 参数 (从 零 开 始 的 整 型 ), 其 返回 值 是 jQuery。 参数 callback (Function) 


了 。360 。 
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是 对 于 每 个 匹配 的 元 素 所 要 执行 的 函数 。 如 果 返 回 false， 则 停止 循环 (类 似 break); 如 果 返 加 
true， 则 跳 转 至 下 一 个 循环 〈 类 似 continue)。 
示例 : 友 代 两 个 图 像 ， 并 设置 它们 的 src 属性 。 代 码 如 下 : 
<img/><img/> 
$("img").each(function(i){ 
this.src = "bgcolor" +1+".gif"; 


<img strc=" bgcolor 0.gif' />, <img src=" bgcolor 1.gif' /> | 


说 明 : | 
本 示例 中 的 this 指 的 是 DOM 对 象 而 不 是 jQuery 对 象 | 


(7) text0 取 得 所 有 匹配 元 素 的 内 容 。 
结果 是 由 所 有 匹配 元 素 包 含 的 文本 内 容 组 合 起 来 的 文本 。 该 方法 对 HTML 和 XML 文档 均 | 
有 效 。 | 
应 用 示例 
本 示例 将 应 用 jQuery 完成 对 XML 的 读 取 。 其 具体 步 又 如 下 : | 
(1) 创建 index.php 文件 ， 调 用 jQuery 框架 包 ， 并 且 直接 编写 JavaScript 脚本 文件 ， 应 用 ， 
jQuery 中 的 函数 完成 对 指定 XML 文件 的 读 取 ， 通 过 setTimeout0 函 数 每 间隔 1000 毫秒 执行 一 | 
次 函数 ， 实 现 对 XML 文件 内 容 的 动态 读 取 。 其 关键 代码 如 下 : 
<script src="js/jquery-1.3.2.js" type="text/javascript"></script> 
</head> 
<script type="text/javascript"> 
S$(document).ready(function ReadXMIO{ 
$.get(Rss.xml', function(d){ 
$('body").emptyO; // 清 空 数据 | 
$('body”).append('<h1> JQuery 动态 读 取 XMI 文件 ，<a hre 伟 "Rssxml" class="STYLE3"> | 
查看 XML 文件 内 容 </a></h1>"); | 
$Cbody).append(<dl />"): /添加 标签 
$(d).find('item').each(functionO{ //item 元 素 
var $item = $(this); 
var title = $item.find('title').text0; /获取 title 的 值 | 
varlinks = Sitem find(link)text0; /获取 link 的 值 | 
Var description = $item.find('description').textO; | 
var pubDate = $item find(‘pubDate').textO: | 
Var html ='<table width="672" border="1" cellpadding="1" cellspacing="1" bordercolor= 
"#FFFFFF" bgcolor="#999999"><tr>"; 
html +='<td width="102" height="22" align="right" bgcolor="#FFFFFF"> 标 题 : </td>' 
html + 一 <td width="368" bgcolor="#FFFFFF">&nbsp;'+titlet'</td>"; | 
html :+= '<td width="184" align="center” bgcolor="#FFFFFF"> 时 间 : ， 


'+pubDatet'</td>"; 
html += '</tr><tr>"; 
html += '<td colspan="3" height="40"; bgcolor="#FFFFFF">&nbsp;&nbsp;'+ 
descriptiont+'</td>"; 
html += "</tr><tr>"; 
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html += '<td height="25" align="right" bgcolor="#FFFFFF">URL:</td>'; 
html += '<td colspan="2" bgcolor="#FFFFFF">&nbsp:;&nbsp;'+links+"'</td>'; 
html += "</tr></table>"; 

html +="</table>"; 


$('dl").append($(htm))); /向 中 添加 数据 
BD 
DE 
setTimeout("ReadXMIO", 1000); /间隔 1000 毫秒 执行 一 次 ReadXML 函数 
| | 让 
| </script> 


(2) 创建 Rss.xml 文件 ， 存 储 于 示例 的 根 目录 下 。 
运行 结果 如 图 11.8 所 示 。 
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| 专家 点 评 

在 实现 XML 文件 的 动态 读 取 时 , 为 了 能 够 获取 到 XML 文件 中 最 新 的 数据 , 必须 对 Intemet 
的 临时 文件 进行 如 图 11.9 所 示 的 设置 。 其 操作 步骤 是 : 在 正 浏览 器 中 选择 “工具 ”一 “Internet 
选项 ”命令 ， 在 Intemet 临时 文件 中 单 击 “ 设 置 ” 按 钮 ， 在 弹出 的 设置 对 话 框 中 选中 “每 次 访 
问 此 页 时 检查 ” 单 选 按钮 ， 然 后 单 击 “ 确 定 ”按钮 。 
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问题 232 
问题 233 
问题 234 
问题 235 
问题 236 
问题 237 


如 何 创 建 简单 的 PDF 文档 ? 

如 何在 PDF 中 插入 图 片 ? 

如 何 设置 PDF 文档 的 页 眉 与 页 脚 ? 
如 何在 PDF 文档 中 绘制 表格 ? 

如 何在 PDF 文档 中 输出 中 文 ? 

如 何 实现 PDF 文档 下 载 ? 


流 
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问题 232 ”如 何 创 建 简单 的 PDF 文档 ? 


侠 】 ， 占 题 阐述 
如 何 创 建 简单 的 PDF 文档 ? 
| 专家 解答 
| 通过 FPDF 可 以 创建 PDF 文档 ， 其 实现 主要 过 程 为 : 首先 要 实例 化 FPDF 类 ， 然 后 调用 类 
| 中 的 函数 (Open 函数 、AddPage 函数 、SetFont 函数 、Cell 函数 和 Output 函数 ) 完成 文档 的 创建 。 
| (1) 创建 FPDF 对 象 的 语法 格式 如 下 : 
$pdf = new FPDF([string page-orientation [, string measure-unit [, string page-format]]]) 
FPDF 对 象 的 参数 说 明 如 表 12.1 所 示 。 


表 12.1 FPDF 对 象 的 参数 说 明 


参 数 说 了 明 
age-orientation | 设置 PDF 文档 的 排版 ， 可 选 值 为 : P: 表示 竖 向 ; L: 表示 横向 
measure-unit 设置 文档 中 位 置 的 计量 单位 。 可 选 值 为 : pt: 点 ， mm: 毫米 ，cm: 厘米 ，in: 英寸 
设置 PDF 文档 纸张 的 类 型 。 可 选 值 : 表示 纸张 类 型 的 字符 串 , 例如 “A4”、“A5”、 
page-format 


“Letter”， 或 者 一 个 包含 有 两 个 元 素 的 二 维 数组 ， 直 接 指明 纸张 的 大 小 


(2) Open 函数 用 于 标示 开始 创建 PDF 文档 ， 语 法 格式 如 下 : 
$pdf->OpenO 
(3) AddPage 函数 为 PDF 文档 添加 一 个 新 页 ， 语 法 格式 如 下 : 
$pdf ->AddPage([string page-orientation]) 
| 其 中 , page-orientation 表示 创建 的 PDF 文档 是 横向 还 是 竖 向 的 。 与 前 面 介绍 的 创建 对 象 时 
| 的 使 用 方法 相同 。 
(4) SetFont 函数 设置 当前 使 用 的 字体 ， 语 法 格式 如 下 : 
S$pdf->SetFont(string font [, string style [, float size]]): 
| 其 中 ，font 表示 字体 ; style 表示 样式 ， 它 的 可 选择 值 有 如 下 3 种 (如 果 没 有 指定 style 的 
| 值 ， 则 默认 为 普通 的 形式 ): 
| B: 粗 体 
I: 斜体 
U: 下 滑 线 
size 表示 字体 的 大 小 。 如 果 不 指定 ， 则 取 默 认 值 ， 为 12pt。 
(5) Cell 函数 用 于 为 当前 PDF 文档 增加 一 个 单元 格 ， 其 语法 格式 如 下 : 
$pdf->Celldloat w [ float h [, string txt [, mixed border [, int In [, string align [, int fill [, mixed link]]]]]) 
Cell 函数 的 参数 说 明 如 表 12.2 所 示 。 
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表 12.2 ”Cell 函数 的 参数 说 明 


表示 单元 格 宽度 

表示 单元 格 高 度 

表示 单元 格 中 添加 的 文本 字符 串 

表示 单元 格 的 边框 , 当 设 置 为 0 时 表示 无 边框 , 设置 为 1 时 表示 有 边框 ; 或 者 工 表示 左边 框 ， 
T 表 示 顶 部 边框 ，R 表示 右边 框 ，B 表示 底部 边框 

表示 换行 后 ， 设 置 目前 的 位 置 。 可 选 值 为 : 0 表示 向 右 移 动 ; 1 表示 定位 到 下 一 行 的 开始 ; 2 
表示 向 下 移动 。 默 认 值 为 0 | 
设置 字符 串 的 对 齐 方式 。 可 选 值 为 : 工 表示 左 对 齐 ; C 表示 居中 ; R 表示 右 对 齐 。 默 认 值 为 工 
设置 单元 格 的 背景 。 可 选 值 为 : 0 表示 透明 ; 1 表示 不 透明 。 默 认 值 为 0 
设置 一 个 URL 链接 或 由 AddLink 函数 添加 的 内 部 链接 


(6) Output 函数 用 于 输出 PDF 文档 ， 语 法 格式 如 下 : 
Spdf->Output([string filename [, bool download]]) 
其 中 ,filename 表示 要 存储 的 文件 名 。 如 果 不 指定 文件 名 ， 则 浏览 时 会 在 正 Tr 
打开 ; download 表示 是 否 一 定 要 用 户 下 载 查 看 ， 默 认为 False。 
应 用 示例 
下 面 应 用 上 述 介绍 的 函数 创建 一 个 PDF 文档 ， 输 出 一 段 英文 字符 串 。 
examplel.php 的 代码 如 下 : 


<2php 

define(FPDF FONTPATH'font): /定义 font 文件 夹 所 在 路 径 

require_once('pdf/fpdf.php"): // 包 含 FPDF 类 库 文件 

$pdf=new FPDF('P', mm', 'A4"); 。 // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 毫米 ， 纸 张大 小 为 A4 
S$pdf->Open(O; /开始 创建 PDF 

$pdf>AddPage0): /增加 一 页 

$pdf->SetFont(Courier,T,20); 。 ”// 设 置 字体 样式 

S$pdf->Cell(0,0,'mrsoft!"); // 增 加 一 个 单元 格 

$pdf->Output(): /输出 PDF 到 浏览 器 

?> 


运行 结果 如 图 12.1 所 示 。 

EB rp 168 1 96/7M/ pat/pderamplel pnp - Wndows reme [= 
OO le emoens -| [x]| 
[Es es 本 "” 
从 电 昌国 |C all 加- ii 了 ， 闪 字 
百 = 


mrsoft! 


图 12.1 简单 的 PDF 文档 
技巧 : | 
上 面 的 示例 是 将 PDF 文档 的 内 容 在 正 浏览 器 中 输出 ， 用 户 可 以 对 上 述 代码 中 的 
“$pdf->Output();” 进 行 修改 ， 代 码 为 “S$pdf->Output('mrsoft.pdf,true);”， 定 义 PDF 文档 名 称 
mrsoft.pdf， 设 置 用 户 下 载 查看 。 如 此 即 可 将 PDF 文档 保存 到 服务 器 并 提示 用 户 下 载 。 


“365。 


3 人 Pp 六 预 知道 的 300 个 问题 


| 专家 点 评 


| FPDF 是 一 个 免费 的 PDF 操作 类 , 通过 它 可 以 完成 基本 的 PDF 创建 功能 , 并 且 它 还 支持 中 
| 文 (需要 对 相应 字体 进行 配置 )。FPDF 类 中 常用 的 PDF 操作 函数 包括 : Open 函数 、AddPage 
鲜 | 函数 、SetFont 函数 、Cell 函数 和 Output 函数 。 
FPDE 的 下 载 可 以 通过 访问 http://www.fpdf.org/ 来 得 到 ,下载 完 成 后 将 下 载 的 文件 直接 解压 
| 到 某 一 文件 夹 中 即 可 。 


问题 233 如 何人 在 PDF 中 插入 图 片 ? 


| 问题 阐述 
| 如何 在 PDF 中 插入 图 片 ? 
| 专家 解答 
在 FPDF 中 通过 Image 函数 向 PDF 中 插入 图 片 ， 其 语法 格式 如 下 : 
TImage(string file, float x, float y, float width, float height) 
其 中 ，x 和 yy 表示 图 片 所 在 的 坐标 ，width 和 height 表示 图 片 的 宽度 和 高 度 。 


应 用 示例 
下 面 应 用 Image0 函 数 向 PDF 文档 中 插入 一 幅 图 片 , 并 且 将 PDF 在 正 浏览 器 中 直接 输出 。 
example2.php 的 代码 如 下 : 

<2php 


define(FPDF_FONTPATH','font/); /定义 font 文件 夹 所 在 路 径 
require_once('pdf/fpdf.php"):; // 包 含 fpdf 类 库 文件 
$pdf=new FPDF('P', mm', 'A4"):; 。// 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 毫米 ， 纸 张大 小 为 A4 


S$pdf->Open(); // 创 建 PDF 
S$pdf->AddPage(); /增加 一 页 
S$pdf->SetFont('Courier',T'.20): /设置 字体 样式 
S$pdf->Image('images/as.jpg',10,10,80,80): // 插 入 图 片 
S$pdf->OutputO; // 输 出 PDF 到 浏览 器 
?> 


运行 结果 如 图 12.2 所 示 。 


@ htip//1921681.96/TM/pdi/pdf/exampie2php - Windows Interne.. loss 


OO Bre/ -19[x]|p az EE 
宫 = 帮 htpy/192.168.196/TM/pdf/pdi/example2.php 


入 让 ml jal I LE 


@ ER| Ep 机 式 二 从 = 


图 12.2 在 PDF 中 插入 图 片 


了 。366 。 


第 12 章 PPDF 技 术 一 $9 | 


问题 234 如何 设 置 PDF 文档 的 页 眉 与 页 脚 ? 


问题 阐述 
如 何 设置 PDF 文档 的 页 眉 与 页 脚 ? 
专家 解答 


在 FPDF 中 ， 页 眉 与 页 脚 是 通过 重 写 FPDF 类 中 的 Header 方法 和 Footer 方法 来 实现 的 。 | 
因为 FPDF 类 中 的 Header 方法 和 Footer 方法 虽然 存在 ， 但 是 方法 体 中 没有 任何 内 容 ， 所 以 设 | 
置 PDF 文档 的 页 眉 与 页 脚 需要 重 写 FPDF 类 中 的 Header 方法 和 Footer 方法 。 | 

应 用 示例 | 

下 面 首先 创建 PDF 类 ,继承 FPDF 类 ,在 PDF 类 中 对 Header 方法 和 Footer 方法 进行 重 写 ，| 
完成 页 眉 与 页 脚 的 设置 。 然 后 ， 创 建 PDF 文档 ， 输 出 一 行 英文 字符 串 和 一 幅 图 片 。 | 

example3.php 的 代码 如 下 : 


<?php | 
define(FPDF FONTPATH'font/: /定义 font 文件 夹 所 在 路 径 | 
require_once('pdf/fpdf.php"): // 包 含 FPDF 类 库 文件 | 
class PDF extends FPDF{ /继承 FPDF 类 ， 重 写 页 眉 与 页 脚 的 方法 | 
function HeaderO{ /设置 页 眉 | 
$this->SetFont('Arial"B',15); /设置 页 眉 字 体 | 
$this->Wiite(10.Title): // 写 入 页 眉 文字 | 
S$this->Ln(20); // 换 行 | 
} | 
function FooterO{ /设置 页 脚 | 
Sthis->SetY(-15); // 设 置 页 脚 所 在 位 置 | 
Sthis->SetFont(Arial,T,8): // 设 置 页 脚 字体 | 
S$this->Cell(0,10,Page - ".$this->PageNo0); /输出 当前 页 码 作为 页 脚 内 容 | 
} | 
} | 
$pdf=new PDF(P', mm', 'A4"); 。“”// 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 毫米 ， 纸 张大 小 为 A4 | 
S$pdf->OpenO; // 创 建 PDF 
S$pdf->AddPage(): /增加 一 页 


$pdf->SetFont(Courier,T,20); 。“”// 设 置 字体 样式 
$pdf->Cell(30,30,MRSOFT!"): 。“// 输 出 字符 串 


S$pdf->Image('images/as.jpg',30,50,80.80): // 插 入 图 片 
S$pdf->OutputO); // 输 出 PDF 到 浏览 器 | 
?> | 
运行 结果 如 图 12.3 所 示 。 | 
专家 点 评 


在 设置 PDF 的 页 眉 与 页 脚 时 ， 可 引入 Write、Ln、PageNo 和 SetY 4 个 方法 。 
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B http//192168 .1.96/TM/ pl/pel /example3 Phe ~ Windows Interner Explorer 
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| 其 语法 格式 如 下 
| a 


参数 h 定义 字符 串 的 行列 高 度 ; 参数 txt 指定 输出 的 字符 串 ， 参 数 link 设置 链接 的 网 页 或 


| 者 AddLink0 方 法 的 标识 符 。 


图 12.3 设置 PDF 的 页 眉 和 页 脚 
(1) Write 方法 用 于 输出 字符 串 。 当 到 达 文 档 的 右边 位 置 或 者 遇 到 〈m) 时 会 自动 换行 。 


(2) In 方法 用 于 换行 操作 。 其 语法 格式 如 下 : 


Ln([float h]) 


参数 h 用 于 设置 行 的 高 度 。 默 认 值 为 最 后 输出 行 的 高 度 。 
(3) PageNo 方法 的 作用 是 返回 当前 页 码 。 其 语法 格式 如 下 : 


int PageNo(); 


(4) SetY 方法 用 于 设置 页 脚 的 坐标 。 其 语法 格式 如 下 : 


SetY (float y); 


| 参数 y 为 页 面 上 的 YY 坐标， 单位 为 毫米 。 如 果 y 为 负数 ， 则 表示 从 页 面 底 部 向 上 的 距离 。 
| 例如 ， 上 面 的 示例 中 设置 y 的 值 为 -15， 即 页 脚 所 在 位 置 距 页 面 底部 15 毫米 。 


问题 235 如 何在 PDF 文档 中 绘制 表格 ? 


问题 阐述 
如 何在 PDF 文档 中 绘制 表格 ? 
专家 解答 


应 用 示例 


通过 FPDF 绘制 表格 的 方法 与 输出 文字 的 方法 类 似 , 在 输出 文字 时 往往 指定 Cell 方法 的 边 
框 参数 为 0， 即 不 输出 边框 。 在 绘制 表格 时 往往 将 其 设置 成 大 于 0 的 整数 ， 用 于 表示 边框 ， 这 
就 是 PDF 文档 中 表格 的 单元 格 。 当 把 这 些 单元 格 组 合 到 一 起 后 就 构成 了 表格 。 


下 面 在 PDF 中 创建 一 个 表格 。example4.php 的 代码 如 下 : 
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<?php 
define(FPDF FONTPATH'font/); /定义 font 文件 夹 所 在 路 径 
require_once('pdf/fpdf.php"): // 包 含 FPDF 类 库 文件 
class PDF extends FPDF{ 
fonction HeaderO{ // 设 置 页 眉 
S$this->SetFont('Arial','B',15); // 设 置 页 眉 字体 
S$this->Write(10,'Pay Slip’); // 写 入 页 眉 文字 
S$this->Ln(20); /换行 
} | 
function FooterO{ // 设 置 页 脚 | 
S$this->SetY(-15); // 设 置 页 脚 所 在 位 置 | 
Sthis->SetFont('Arial,T',8):; // 设 置 页 脚 字体 | 
Sthis->Cell(0.10.Page - ,Sthis->PageNo0): /输出 当前 页 码 作为 页 脚 内 容 | 
} : 
$pdf=new PDF(P', "mm', 'A4"); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 毫米 ， 纸 张大 小 为 A4 ， 
$pdf->Open0; /开始 创建 PDF | 
$pdf->AddPage(): // 增 加 一 页 | 
$pdf->SetFont( Arial,",14): // 设 置 字体 样式 | 
S$header=array(Name','Age','Duty',Laborage’): /设置 表 头 
$data=array(); /设置 表 体 


$data[0] = array('Simon','24''employee','5,000.00"):; | 
$data[1] = array('Elaine','25',"Manager','6,000.00"); 
$data[2] = array('Susan','25','employee','4,000.00"):; 
$data[3] = array('David','26',"Manager','8,000.00"); 


S$width=array(40,40,40,40); // 设 置 每 列 宽度 
for($i=0;$i<count($header):$i++) // 循 环 输出 表 头 
Spdf->Cell($width[$i],6,$header[$1],1):; 

S$pdf->LnO); 

foreach($data as $row){ /循环 输出 表 体 


$pdf->Cell($width[0].6,$row[0].1): 
S$pdf->Cell($width[1],6,$row[1],1); 
S$pdf->Cell($width[2],6,$row[2],1); | 
$pdf>Cell(Swidth[3].6.$row[3].1: | 


S$pdf->LnO; 
} 
$pdf->OutputO; /输出 PDF 到 浏览 器 
?> 
运行 结果 如 图 12.4 所 示 。 
@ http//192168.196/TM/pdf/pdi/examples.php ~ Window Intermet Explorer (~ © mw 
GO] e/a Weedcoenaette -sx) Par Pp" 
会 Bs | 坊 htpy/192158196/TW/pdi/pdflexample4php | 
国人 | EPE | 
DO| Poysip EF | 
| 2 | 1 
IName |Age Duty Laborage 1 
醒 [Simon 24 employee 5,000 00 1 
Elaine [25 Manager 5,000 00 1 
| Susan 25 |employee 4.000 00 
David 26 Manager 8.00000 
| 
! EC- 


图 12.4 在 PDF 中 绘制 表格 | 
.369 。 


本 人 Pp 六 须知 道 的 300 个 问题 


问题 236 ”如 何在 PDF 文档 中 输出 中 文 ? 


问题 阐述 

若 直 接 通 过 FPDF 在 PDF 文档 中 输出 中 文 , 会 发 现 有 时 输出 的 中 文字 符 串 是 乱码 。 该 如 何 
解决 这 个 问题 ? 
专家 解答 

解决 此 问题 的 方法 是 应 用 FPDF 提供 的 一 个 中 文 插件 ， 继 承 插件 中 的 PDF_Chinese 类 ,在 
PDF 文档 中 输出 中 文 。 

首先 ， 下载 并 解压 此 插件 ， 复制 其 中 的 chinese.php 文件 ， 并 将 其 存储 于 项 目的 指定 文件 夹 
下 ， 然 后 通过 include 语句 在 创建 PDF 文档 的 文件 中 载 入 chinese.php， 最 后 ,创建 PDF 类 ， 继 
承 PDF_Chinese 类 ， 并 在 PDF 文档 中 输出 中 文 。 

应 用 示例 

本 实例 载 入 PDF 中 文 插件 ， 通 过 中 文 插件 在 PDF 中 输出 中 文字 符 串 。 

其 实现 的 example5.php 的 代码 如 下 : 


<2?php 
include('‘pdf/chinese.php'): // 载 入 中 文 插件 
class PDF extends PDF_Chinese{ // 继 承 中 文 插件 类 
function HeaderO{ /设置 页 眉 
S$this->SetFont(GB',",10); // 设 置 字体 
S$this->Write(10,'FPDEF 中 文 测试 "); /设置 页 眉 的 内 容 
S$this->Ln(20); /执行 换行 操作 
} 
function FooterO{ // 设 定 页 脚 
Sthis->SetY(-15); /设置 页 脚 的 输出 坐标 
S$this->SetFont('GB',",10); // 设 置 字体 
S$this->Cell(0,10,' 第 '.$this->PageNo0O.' 页 "); /设置 页 脚 输出 的 内 容 
} 
} 
S$pdf=new PDFO; // 创 建 PDF 文档 
S$pdf->AddGBFont(): /添加 字体 
$pdf->Open(); /开启 文档 
$pdf->AliasNbPages0: /为 每 个 页 面 定义 一 个 别名 
$pdf->AddPage0: /添加 页 
S$pdf->SetFont(GB',T',20): // 设 置 字 体 


$str=' 山 不 在 高 ， 有 仙 则 名 。 水 不 在 深 ， 有 龙 则 灵 。 斯 是 陋室 ， 惟 吾 德 世 。 苦 痕 上 阶 绿 ， 草 色 入 
廉 青 。 谈 笑 有 鸿儒 ， 人 往来 无 白丁 。 可 以 调 素 雄 ， 阅 金 经 。 无 丝竹 之 乱 耳 ， 无 案 读 之 劳 形 。 南 阳 诸 葛 庐 ， 
西 蜀 子 云 亭 。 和 孔子 云 :“ 何 陋 之 有 ? ”'; 

$pdf>Write(10.$stD: // 输 出 中 文 

$pdf->Output|; 

> 


“370” 


运行 结果 如 图 12.5 所 示 。 
专家 点 评 


在 创建 生成 PDF 文档 的 文件 (example5 
php) 时 , 页 面 的 编码 格式 应 设置 成 gb2312， 
如 果 页 面 的 编码 格式 是 UTF-8, 那么 即使 继 
承 了 PDF_Chinese 类 ， 输 出 的 中 文 仍然 会 
是 乱码 。 


图 12.5 在 PDF 中 输出 中 文 


问题 237 如何 实现 PDF 文档 下 载 ? 


问题 阐述 
如 何 实现 PDF 文档 下 载 ? 
专家 解答 


要 实现 PDF 文档 的 下 载 操作 ， 只 需要 在 通过 Output 函数 输出 PDF 文档 时 添加 两 个 参数 即 
可 ， 第 一 个 参数 定义 下 载 的 PDF 文档 的 名 称 ， 第 二 个 参数 设置 为 me， 提醒 用 户 下 载 查看 ， 如 果 | 
将 第 二 个 参数 设置 为 false, 那么 生成 的 PDF 文档 会 直接 存储 于 服务 器 中 , 不 会 提醒 用 户 下 载 查 看 。 | 
应 用 示例 
下 面 对 示 例 example5.php 进行 修改 ， 将 其 生成 的 PDF 文档 命名 为 “程序 测试 .pdf”， 并 且 | 
提醒 用 户 下 载 查 看 。 修 改 的 关键 代码 如 下 : | 


<2php 

include('pdf/chinese.php'): // 载 入 中 文 插件 

/省 略 了 部 分 代码 | 

$str=' 山 不 在 高 ， 有 仙 则 名 。 水 不 在 深 ， 有 龙 则 录 。 斯 是 陋室 ， 惟 吾 德 又 。 蔡 痕 上 阶 绿 ， 草 色 入 | 
廉 青 。 谈 笑 有 鸿 侍 ， 往 来 无 白丁 。 可 以 调 素 琴 ， 阅 金 经 。 无 丝竹 之 乱 耳 ， 匹 案 读 之 劳 形 。 南 阳 诸 葛 访 ， | 
西 蜀 子 云 亭 。 和 孔子 云 :“ 何 陋 之 有 ? ” 

S$pdf->Write(10,$str): /| 输出 中 文 

$pdf->Output( 程 序 测试 .pdf,true); 


?> 


修改 后 的 示例 存储 于 example6.php 文件 中 ， 运 行 此 示例 将 输出 如 图 12.6 所 示 的 PDF 文件 
下 载 页 面 。 | 


Er 3 


您 起 打开 或 保存 此 文件 归 ? 
| 宇 站 人 
人 
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图 12.6 PDF 文 档 下 载 
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问题 238 
问题 239 
问题 240 
问题 241 
问题 242 
问题 243 
问题 244 
问题 245 
问题 246 
问题 247 


向 Flash 中 传递 参数 
如 何 通过 embed 标签 播放 音乐 ? 
如 何 通过 object 标签 播放 音乐 ? 
如 何 实现 在 线 音 乐 中 歌词 同步 输出 ? 
什么 才 是 音乐 在 线 播放 的 关键 ? 
(关闭 网 页 时 ) Ajax 无 刷新 删除 .m3u 格式 的 文件 
如 何 将 选 定 的 音乐 添加 到 播放 列表 中 ? 
如 何 判 断 在 线 音乐 文件 的 地 址 是 否 正确 ? 
如 何在 网 页 中 播放 FLV 文件 ? 
如 何 实现 MP3 在 线 点 播 ? 


需 
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问题 238 ”向 Flash 中 传递 参数 
问题 阐述 | 
在 Web 项 目 开 发 中 ,通过 Flash 动画 来 打造 一 些 特殊 的 网 页 效果 是 很 常见 的 方法 。 其 中 所 
用 动画 多 数 是 单纯 的 动画 效果 ， 与 项 目 中 的 数据 联系 不 是 很 紧密 ， 那 么 是 否 可 以 将 动画 中 的 内 | 
容 与 项 目 紧 密 地 结合 起 来 呢 ? 根据 动画 中 内 容 的 变换 ， 而 改变 项 目 中 输出 的 内 容 呢 ? 这 就 是 我 | 
们 将 要 解决 的 问题 ， 根 据 Flash 中 传递 的 参数 ， 输 出 不 同 的 网 页 内 容 。 
专家 解答 


向 Flash 中 传递 参数 的 具体 步骤 如 下 : | 
(1) 根据 项 目 中 的 需要 创建 Flash， 并 且 向 Flash 中 传递 参数 值 。 在 Flash 元 件 中 通过 | 
on(release) 方 法 获取 传递 的 参数 值 ， 并 执行 超 链接 。 
通过 this.loaderinfo.parameters["conn_type"] 获 取 传 递 的 参数 值 。 
var type=this.loaderinfo.parameters["conn_type"]; 
Flash 的 中 的 编辑 结果 如 图 13.1 所 示 。 


wr :声明 局 部 到 量 
要 量 (W: | Vpe = thissloaderinio, parameter st'conn type’] ] 
RD): | 人 到 > > 口 为 型 二 加 mport 动作 

虫口 四 日 v [和 有 于 “| 四 
om (relesse) T 四 

有 如 时 要 L( peLleft .php?pomn_it=205101114tab1n_nane=studysclass_+ype= 从 地 开始 tcenr_type=tyne", "leftvindow "ET"1 : 国 
1 


i 和 
图 13.1 通过 变量 传递 参数 
通过 getURL 方法 执行 超 链接 。 


getURL("pd left php?conn id=20510111&table name=study&zclass_type= 从 零 开 始 &conn type=type"，， 
"leftwindow", "GET"): 


Flash 中 的 编辑 结果 如 图 13.2 所 示 。 


| etURL ; 通知 Web 浏 此 加 定位 脂 定 的 URL 


UR (UW): | pd_left ohp?cann jo=2051011 18table_name=tudydclass ype= 从 鹤 开 始 Bcom_ype=type 


Dt) 
OD): Rnd Rt 
秋明; 用 5 二 这 加 
康王 | 有 IDi> = mzyF |@ 
上 Griewe T 人 
a 加 


元 件 1| 厨 


第 3 人行; getURL["pd_jefi.php?conn_HH-20510111tabe_name-studyclass_type- 从 零 开 始 conn_type-type", "eftwndow", "GET 
图 13.2 通过 getURL 执行 超 链接 


此 处 在 Flash 中 定义 了 3 个 元 件 ， 分 别 对 应 入 门 、 开 发 和 应 用 3 个 模块 。 在 每 个 Flash 元 
件 中 分 别 获取 对 应 的 参数 值 。 
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元 件 1 的 脚本 代码 如 下 : 
on (release) { 
var type=this.loaderinfo.parameters["conn type"]: 
getURL("pd leftphp?conn id=20510111&table name=study&class type= 从 零 开始 &conn type= 
type", "leftwindow", "GET"); 
} 
元 件 2 的 脚本 代码 如 下 : 
on (release) { 
var type=this.loaderinfo.parameters["conn type"]; 
getURL("pd left.php?conn id=10121002&table name=jszx&class_type= 技 术 中 心 &conn type= 
type", "leftwindow", "GET"); 
} 
元 件 3 中 的 脚本 代码 如 下 : 
on (release) { 
var type=this.loaderinfo.parameters["conn_type"]; 


getURL("pd_left.php?conn id=10127105&table_ name=ymzx&cclass type= 源 码 管理 &conn type= 
type", "leftwindow" "GET"): 


} 
(2) 在 项 目 中 创建 js 脚本 文件 flashjs， 定 义 向 Flash 中 写 入 的 内 容 。 其 代码 如 下 : 
function flash(m) { 


var flash2='<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" codebase= 
"http://download macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version=7,0,19,0" width="171" height="93">"; 
© flash2+="<param id="swf" name="movie" value="images/bccd.swf?conn type=+m+">"; 
flash2+='<param name="quality" value="high">"; 
flash2+='<param name="allScriptAccess" value="always"/>"; 
flash2+='<param name="wmode" value="transparent">"; 
flash2+='<embed src="images/bccd.swf" width="171" height="93" quality="high" pluginspage= 
"http://www.macromedia.com/go/getflashplayer" type="application/x-shockwave-flash" wmode="transparent"> 
</embed>"; 
flash2+="</object>"; 
四 document.write(flash2):; 
) 


代码 导读 : 
@ param id="swf"': 在 swf 的 url 中 传递 参数 值 。 
@ document.write: 执行 写 入 操作 。 
(3) 在 项 目的 指定 文件 中 载 入 定义 的 js 脚本 文件 ， 调 用 定义 的 flash() 方 法 输出 Flash 
动画 。 
<script sre="js/flash.js"></script> 
<script type="text/javascript"> 
flash('{$conn type}"); 
</script> 


在 本 程序 的 LOG 中 应 用 Flash 实现 入 门 、 开 发 和 应 用 3 个 模块 之 间 的 跳 转 ， 并 根据 Flash 
. 374: 


第 13 章 PHP 与 多 如 体 技术 


中 传递 的 参数 ， 在 项 目 中 完成 不 同 内 容 之 间 的 切换 操作 ， 其 运行 结果 如 图 13.3 所 示 。 
入 门 应 用 
-Es 
图 13.3 通过 Flash 完成 模块 之 间 的 跳 转 


问题 239 如何 通过 embed 标签 播放 音乐 ? 


问题 阐述 
在 Web 项目 中 , 经 常会 看 到 支持 在 线 音乐 播放 功能 的 网 站 , 那么 它们 的 音乐 在 线 播放 功能 
是 如 何 实现 的 呢 ? 这 里 我 们 将 介绍 众多 方法 中 的 一 种 ， 即 通过 embed 标签 实现 音乐 在 线 播放 。 | 
专家 解答 
<embed> 是 html 标签 中 的 播放 多 媒体 标签 ， 通 过 它 可 以 实现 音乐 的 在 线 播 放 功能 。embed | 
标签 的 基本 语法 如 下 : 
<embed src=url> 
Wrl: 为 音频 或 视频 文件 的 路 径 ， 可 以 是 相对 路 径 ， 也 可 以 是 绝对 路 径 。 
embed 标签 可 以 用 来 播放 各 种 多 媒体 文件 ， 格 式 可 以 是 midi、wav、mp3 等 ， 目 前 主流 的 
浏览 器 一 般 都 支持 该 标签 。 
embed 标签 有 很 多 属性 ， 常 用 的 属性 及 说 明 如 表 13.1 所 示 。 
表 13.1 embed 标签 常用 属性 及 说 明 


该 属性 规定 音频 或 视频 文件 是 否 在 下 载 完 之 后 就 自动 播 
autostart | 放 。 当 属性 值 为 tue 时 ， 音 乐 下 载 完 成 后 自动 播放 ， 当 属 |<embed src="1.mp3" autostart=true> 
性 值 为 ftlse 时 ， 下 载 完成 后 不 播放 

该 属性 规定 音频 或 视频 文件 是 否 循环 及 循环 次 数 。 当 属性 
loop 值 为 tue 时 ， 音 频 或 视频 文件 循环 : 当 属 性 值 为 false 时 ，|<embed src="1.mp3" loop=true> 
音频 或 视频 文件 不 循环 。 如 果 为 正 整数 ， 则 为 循环 次 数 
该 属性 规定 控制 面板 是 否 显示 ， 默 认 值 为 no。 当 属性 值 为 
tmue 时 ， 隐 藏 面板 ， 当 属性 值 为 no 时 ， 显 示 面 板 

该 属性 规定 音频 或 视频 文件 开始 播放 的 时 间 。 如 果 未 定义 
则 从 文件 开头 播放 

该 属性 规定 音频 或 视频 文件 的 音量 大 小 。 如果 未 定义 则 使 
用 系统 本 身 的 设 定 

width “| 该 属性 规定 了 控制 面板 的 宽度 

height “| 该 属性 规定 了 控制 面板 的 高 度 


hidden <embed src="1.mp3" hidden="yes"> | 


starttime <embed src=- "1.mp3" startime="00:10"> 


volume <embed src="1.mp3" volume="10"> 


<embed src="1.mp3" width="100"> 
<embed src="1.mp3" height="200"> 
<embed src="l.mp3" title="a good 


title 该 属性 规定 音频 或 视频 文件 的 说 明文 字 
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千 

区 
除了 上 述 属性 外 ,embed 标签 还 可 以 对 面板 的 外 观 进行 设置 , 包括 面板 的 背景 色 和 前 景色 等 。 
应 用 示例 


| 播放 存储 于 upfile 文件 夹 下 的 :mp3 文件 ， 其 代码 如 下 : 
| <embed src="upfile/brzy. mp3" align="middle" ShowStatusBar=true></embed> 
专家 点 评 
通过 embed 标签 不 但 可 以 实现 音乐 的 在 线 播放 ， 在 充分 发 挥 loop 属性 作用 的 前 提 下 ， 还 
| 可 以 实现 在 线 音乐 的 循环 播放 和 连续 播放 。 其 具体 原理 是 : 通过 sre 属性 来 获取 播放 文件 的 路 
| 径 ; 判断 当 loop 的 属性 值 为 tue 时 执行 循环 播放 ; 判断 当 loop 的 属性 值 为 false 时 执行 连续 播 
| 放 。 例 如 ， 根 据 表单 提交 的 值 判断 是 执行 循环 播放 还 是 连续 播放 。 其 关键 代码 如 下 : 


| <embed src="upfile/brzy.mp3"” loop="<?php if($_POST['Submit] 一 "连续 播放 "){echo "false";} 
| elseif($ POST['Submit2'] 一 "循环 播放 "){fecho "true":}?>" align="middle" ShowStatusBar=true></embed> 


问题 240 ”如何 通过 object 标签 播放 音乐 ? 


| 问题 前述 


| 能 够 实现 音乐 在 线 播放 的 方法 众多 ， 前 面 已 经 为 读者 介绍 一 种 了 ， 下 面 我 们 再 介绍 一 种 ， 
| 即 通过 object 标签 实现 音乐 在 线 播放 。 

| 专家 解答 

| object 标签 定义 一 个 嵌入 的 对 象 ， 通 过 该 对 象 向 HTML 页 面 添 加 多 媒体 。object 标签 的 基 
| 本 格式 如 下 : 

| <object classid="clsid:22D6F312-BOF6-11D0-94AB-0080C74C7E95" height="68" id="MediaPlayer1" 
| width="460"> 
| <param name="ShowStatusBar" value="-1"> 


<param name="Filename" value="<?php echo $_GET[id]?>"> 
</object> 


classid: 设置 或 获取 对 象 的 类 标识 。 
param: 可 定义 用 于 对 象 的 run-time 设置 。 
应 用 示例 
通过 object 标签 播放 指定 的 在 线 音乐 文件 ， 其 关键 代码 如 下 : 
<!-- 通 过 object 标签 播放 指定 的 文件 -> 
| <object classid="clsid:6BF52AS2-394A-11D3-B153-00C04F79FAA6" id="mediaPlayer" width="480" 
| height="64"> 
| <param name="url" value="upfiles/video/<?php echo $ GET[id]:?>"> /指定 播放 文件 的 路 径 
<param name="volume" value="100"> 
<param name="playcount" value="100"> 
<param name="enablecontextmenu" value="0"> 


<param name="enableerrordialogs" value="0"> 
</object> 
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专家 点 评 

objeet 是 微软 专门 为 匡 浏览 器 打造 的 、 可 以 扩展 外 部 应 用 程序 及 插件 的 对 象 标签 ， 它 和 | 
embed 的 不 同 之 处 在 于 ，object 只 支持 以 匡 技术 为 核心 的 浏览 器 系列 ， 而 对 其 他 的 浏览 无 效 ， |  ， 
如 果 要 使 用 object 来 播放 多 媒体 ， 那么 就 需要 安装 相应 的 播放 插件 ， 如 realplay、quicktime 等 。| 


问题 241 如何 实现 在 线 音乐 中 歌词 同步 输出 ? 


问题 阐述 | 
大 家 经 常会 在 在 线 音乐 网 站 中 看 到 在 播放 音乐 的 同时 ， 对 应 的 歌词 也 会 同步 输出 ， 那 么 这 
个 功能 是 如 何 实现 的 呢 ? 这 里 我 们 就 来 探讨 一 下 这 个 问题 。 | 


专家 解答 


对 于 歌词 同步 功能 的 实现 ， 首 先 要 解决 的 是 。 mmo a 
歌词 的 问题 ， 歌 词 文 件 以 .re 为 后 级 。 在 歌词 文 [6 六 se 
件 中 存储 的 是 在 指定 的 时 间 段 内 所 演唱 的 歌词 。 | :ss 
歌词 文件 中 存储 的 内 容 如 图 13.4 所 示 。 
在 拥有 了 歌词 文件 后 , 就 可 以 通过 PHP 语句 
和 JavaScript 脚本 语句 实现 歌词 同步 显示 。 其 操 jeess 雪 人 
作 步 曙 如 下 : 人 
(1) 应 用 Ajax 技术 封装 一 个 JavaScript 方 Je 2 一 AR 
法 process0， 实 现 歌词 的 无 刷新 读 取 。processO 图 13.4 歌词 文件 中 存储 的 内 容 
方法 的 语法 如 下 : 
/定义 XMLHttpRrequest 对 象 
var XmlHttp=createXmlHttpRequestObjectO; 
/获取 XMLHttpRrequest 对 象 
function createXmlHttpRequestObjectO{ 
// 用 来 存储 将 要 使 用 的 XMLHttpRrequest 对 象 
Var xmlHttp; 
// 如 果 在 Intemet Explorer 下 运行 
if(window.ActiveXObject){ 
try{ 
xmlHttp=new ActiveXObject("Microsoft. XMLHTTP"); 
}catch(e){ 
xmlHttp=false; 


ot enn 如 “王子 制作 


| 
jelse{ 
/如 果 在 Mozilla 或 其 他 的 浏览 器 下 运行 
try{ 
xmlHttp=new XMLHttpRequest(); 
}catch(e){ 
xmlHttp=false; 
9 
} 
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/返回 创建 的 对 象 或 显示 错误 信息 


| 这 !xmlHttp) 
| alert(" 返 回 创建 的 对 象 或 显示 错误 信息 "); 
else 
retum xmlHttp; 
} 
/使 用 XMLHttpRequest 对 象 创建 异步 HITP 请 求 
function process(ids){ 
/在 xmlHttp 对 象 不 忙 时 进行 处 理 


这 xmlHttp readyState 一 4 || xmlHttp.readyState—0){ 
xmlHttp.open("GET","lyric.php?id="+ids,true); 

// 定 义 获取 服务 器 端 响 应 的 方法 
xmlHttp.onreadystatechange=handleServerResponse; 
// 向 服务 器 发 送 请 求 

xmlHttp.send(null); 

}else 

1/ 如果 服务 器 忙 ，1 秒 后 重 试 
setTimeout('process(O)',1000); 


} 
// 当 收 到 服务 器 端的 消息 时 自动 执行 
function handleServerResponseO{ 
/在 处 理 结束 时 进入 下 一 步 操作 
这 xmlHttp readystate 一 4){ 
/状态 为 200 表示 处 理 成 功 结束 
这 xmlHttp.status 一 200){ 
/获取 服务 器 端 发 来 的 XML 信息 
XmlResponse=xmlHttp.TesponseText: 
document.getElementById("lreContent").innerHTML=xmlResponse; 
}else{ 
// 如 果 HTTP 的 状态 不 是 200 表示 发 生 错 误 
alert("There was a problem accessing the server:"+xmlHttp.statusText); 


} 
(2) 定义 异步 请 求 的 lyric.php 文件 ， 应 用 file_get_contents() 函 数 读 取 歌词 文件 中 的 内 容 ， 
在 js 的 handleServerResponse() 方 法 中 将 歌词 文件 中 返回 的 内 容 写 入 ID 等 于 lrcContent 的 标 
| 签 中 。 


(3) 在 音乐 播放 页 中 , 在 播放 音乐 的 同时 调用 process0 方 法 读 取 歌词 内 容 ， 在 div 标签 中 
完成 歌词 的 同步 输出 。 其 关键 代码 如 下 : 


<style> 
#lrcollbox td, #lrcollbox font {font-size: 16px:} 
| #lrcoll td { color:#000000; } 
.lrcbe { overflow:hidden; height:20; width:0: filter:alpha(opacity=100); } 
</style> 
‘<script Src="js/check lyric.js" language="javascript"></script> 
<script> 

window.onunload=functionO{ 

clearTimeout(lrc0): 
} 
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</script> 
<!-- 调 用 process 函数 ， 实 现 无 刷新 读 取 指 定 的 歌词 文件 ， 并 将 内 容 输出 到 lrcContent 中 --> 
<body onLoad="process(brzy")" > 
<center> 
<span id="lrcContent" style="display:none;"><?php include("lyric.php"):?></span> 
<object classid="clsid:6BF52A52-394A-11D3-B153-00C04F79FAA6" id="mediaPlayer" width="420" 
height="64"> 
<param name="url" value="upfiles/video/brzy.mp3"> 
<param name="volume" value="100"> 
<param name="playcount" value="100"> 
<param name="enablecontextmenu" value="0"> 
<param name="enableerrordialogs" value="0"> 
</object> 
<div id="lrcollbox" style="overflow:hidden: height:100; width:420:"> 
<table border="0" cellspacing="0" cellpadding="0" id="lrcoll" width="100%" style="position:relative; | 
top:120px;"> | 
<tr> 
<td nowrap height="20" align="center"> 
<table border="0" cellspacing="0" cellpadding="0"> 
<tr> | 
<td nowrap height="20"><span id="lrcboxl" style="height:20; color:#EF0000"> | 


正在 加 载 歌 词 ……</span></td> ! 
</tr> | 
<tr style="position:relative: top: -20px; Z-index:6:"> | 
<td nowrap height="20"><div id="lrcbcl" class="lrcbc"></div></td> | 
</tr> | 
</table> | 
</td> | 
</tr> ! 
<2php | 
Slyric=file_get_contents("upfiles/lyric/brzy.lre"); | 
$counts=preg_match_all("A[.*?\]/",$lyric,$tmptime):; /提取 []， 判 断 该 行 有 几 个 时 间 | 

for($i=0:$i<$counts:$i++){ 

?> 
<tr style="position:relative; top: <?php echo -20*$1;?>px:;"> 
<td nowrap height="20" align="center"> 

<table border="0" cellspacing="0" cellpadding="0"> | 
<tr><td nowrap height="20"><span id="lrcbox<?php echo $i+2;?>" style= | 
"height:20"></span></td></tr> | 
<tr style="position:relative: top: -20px: z-index:6:"><td nowrap height="20"><div | 
id="lrcbe<?php echo $i+2;?>" class="lrcbe"></div></td></tr> | 


</table> 
</ltd> 
</tr> 
<?php 
?> 
</table> 
</div> 


(3) 封装 JavaScript 脚本 ， 实 现 歌 词 的 滚动 输出 ， 其 中 涉及 的 JavaScript 代码 可 参考 本 书 


} 
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配套 资源 中 的 内 容 。 
专家 点 评 


歌词 的 同步 输出 关键 在 于 Ajax 无 刷新 技术 的 应 用 , 通过 它 读 取 .lrc 格式 的 歌词 文件 。 注 意 ， 
在 读 取 文本 文件 中 的 歌词 时 应 用 的 是 file_get contentsO 函 数 ， 而 这 个 函数 返回 的 字符 串 是 
gb2312 编码 格式 ， 但 是 Ajax 中 默认 使 用 的 是 UTF-8 编码 格式 ， 所 以 在 输出 歌词 时 需要 应 用 
iconvO 函 数 对 歌词 的 编码 进行 格式 转换 ， 只 有 将 其 转换 为 utf-8 编码 格式 后 才能 够 正常 输出 。 


问题 242 什么 才 是 音乐 在 线 播放 的 关键 ? 


问题 前 述 


音乐 在 线 播 放 功能 的 实现 ， 所 使 用 的 播放 器 、 媒 入 的 多 媒体 标签 ， 都 不 是 最 关键 的 内 容 ， 
最 为 关键 的 是 .m3u 格式 文件 的 创建 ， 通 过 它 实现 音乐 的 在 线 播放 。 
专家 解答 

在 线 播放 功能 的 实现 原理 是 客户 端 提交 歌曲 选项 , 服务 器 端 生 成 .m3u 文件 , 并 且 将 该 文件 
通过 HTTP 协议 下 载 到 客户 端 ， 客 户 端 调用 相应 的 播放 器 在 线 播放 音乐 。 

因此 说 创建 一 个 .m3u 格式 的 文件 是 最 关键 的 。 创 建 .m3u 文件 使 用 的 是 文件 系统 函数 中 的 
fopen() 和 fwriteO) 函 数 以 及 foreach 语句 。 创 建 .m3u 文件 的 操作 步骤 如 下 : 

(1) 应 用 fopen0 函 数 以 添加 模式 打开 服务 器 中 指定 文件 夹 下 的 文件 。 如 果 在 文件 夹 下 不 
存在 此 文件 ， 则 创建 一 个 新 文件 ， 文 件 的 名 称 是 当前 时 间 的 时 间 戳 。 代 码 如 下 : 


<?php 

Su=$_SERVER['HTTP_HOST']: // 获 取 文 件 地 址 
$url="http://"."$u"; // 定 义 访问 地 址 
Sdate=date("His"); /定义 时 间 戳 


Slen=strripos($_SERVER['REQUEST_URIT]."/"): 
$tmp=substr($_SERVER['PHP_SELF'].0.$len): // 获 取 文件 的 根 目 录 
站 (!$fopen = fopen ("upfiles/video/$date.m3u","w")){ // 使 用 添加 模式 打开 文件 ， 文 件 指针 指 在 表 尾 
echo "打开 .m3u 文件 失败 ! "; 
exit ; 
} 
(2) 应 用 foreach 循环 语句 ， 获 取 表 单 中 提交 歌曲 的 ID 值 。 代 码 如 下 : 
foreach($_POST[checkbox] as $key=>$value){// 以 book 数组 做 循环 ， 输 出 键 和 值 
(3) 以 获取 到 的 歌曲 ID 值 为 查询 条 件 ， 从 数据 库 中 读 取 出 指定 歌曲 的 存储 名 称 。 代 码 如 下 : 
$1_sqlstr = "select id,style,name,actor,remark,address from tb_video where id='$value'order by id"; 
$Lrst = $conn->execute($] sqlstr): /以 歌曲 了 为 查询 条 件 进行 查询 
(4) 将 歌曲 在 服务 器 中 存储 的 绝对 路 径 写 入 .m3u 文件 中 ， 并 且 在 绝对 路 径 后 添加 一 个 
“mr” 分 隔 符 。 代 码 如 下 : 
这 Ifwrite ($fopen,"".$url.$tmp."/".$1 rst->fields[5].""){ 
print " 写 入 内 容 失 败 !" ; 
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exit ; 
} 
if (!fwrite ($fopen nm)f /添加 不 同 歌曲 绝对 路 径 之 间 的 分 隔 符 
print " 写 入 内 容 失 败 ! "; 
exit; 
} 
} 
es 


至 此 , .m3u 文件 创建 完成 , 接 下 来 就 可 以 通过 embed 标签 来 读 取 .m3u 文件 中 存储 的 歌曲 了 。 


<embed src="upfiles/video/<?php echo $date;?>.m3u" loop="<?php 这 $_ POST[Submitl 一 "连续 播放 ") | 
{echo "false";}elseif($ POST[Submit2] 一 "循环 播放 "){echo "true";}?>"” align="middle" ShowStatusBar= | 
true></embed> | 
专家 点 评 


在 线 音乐 播放 时 创建 的 .m3u 文件 存储 于 服务 器 中 指定 的 目录 下 , 如 果 在 网 站 中 同时 有 很 多 | 
人 上 听 音乐 ， 那 么 就 会 创建 很 多 的 .m3u 文件 ， 当 这 些 人 离开 后 ， 如 果 .m3u 文件 仍然 留 在 服务 器 | 
的 指定 目录 下 ， 会 占用 大 量 的 服务 器 空间 ， 最 终 导致 服务 器 空间 被 耗 尽 。 因 此 必须 在 听 歌 的 用 | 
户 离开 之 后 直接 删除 .m3u 文件 ， 这 样 就 避免 了 服务 器 空间 的 不 必要 的 浪费 。 | 


问题 243 (关闭 网 页 时 ) Ajax 无 刷新 删除 .m3u 
格式 的 文件 


问题 曾 述 | 
既然 已 经 意识 到 .m3u 格式 文件 会 对 服务 器 构成 威胁 ， 那 么 我 们 就 必须 找 出 一 个 解决 的 方 

案 , 如 何 快速 有 效 地 在 浏览 者 离开 网 站 之 后 直接 删除 .m3u 格式 的 文件 , 这 就 是 我 们 这 里 要 解决 | 
的 问题 。 | 
专家 解答 
删除 .m3u 文件 是 通过 onUnload 事件 调用 自 定义 函数 channel checkO 完 成 的 。 在 自 定 义 函 

数 channel checkO 中 ， 应 用 Ajax 无 刷新 技术 调用 delete_ m3u.php 文件 实现 删除 .m3u 文件 的 操 | 
作 。 其 操作 的 步骤 如 下 : | 
(1) 在 播放 页 面 的 body 标签 中 ， 通 过 onUnload 事件 调用 channel_check0 函 数 ， 参 数值 ， 

是 .m3u 文件 在 服务 器 中 的 存储 位 置 。 | 

<body onUnload="channel_check('<?php echo "upfiles/video/".$date.".m3u"; ?>")"> 


(2) 创建 channel checkO 自 定义 函数 ， 应 用 Ajax 无 刷新 技术 ， 调 用 delete_ m3u.php 文件 ， 
实现 删除 .m3u 文件 的 操作 。 其 关键 代码 如 下 : | 
function channel check(names){ 
if(xmlHttp.readyState 一 4 || xmlHttpreadyState 一 0){ /在 xmlHttp 对 象 不 忙 时 进行 处 理 
xmlHttp.open("GET","delete m3u.php?channel name="+names,true); 
xmlHttp.onreadystatechangehandleServerResponse;// 定 义 获取 服务 器 端 响 应 的 方法 
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xmlHttp.sendnulD: /向 服务 器 发 送 请 求 
Tetur false; 
}else 
setTimeout('channel check(Q)',1000); // 如 果 服 务 器 忙 ，1 秒 后 重 试 
} 
function handleServerResponseO{ // 当 收 到 服务 器 端的 消息 时 自动 执行 
if(xmlHttp.readystate—4){ 
if(xmlHttp.status—200){ // 状 态 为 200 表示 处 理 成 功 结束 
xmlResponse=xmlHttp.responseText; /获取 服务 器 端 发 来 的 XML 信息 
Jelse{ 


alert("There was a problem accessing the server:"+xmlHttp.statusText); 
} 
} 
} 


(3) 在 delete_ m3u.php 文件 中 ， 通 过 unlink() 函 数 删除 服务 器 中 指定 文件 夹 下 的 .m3u 文 


| 件 。 代 码 如 下 : 


<?php 

header('Content-Type: text/html; charset=utf-8" ); 

/根据 从 客户 端 获 取 的 用 户 创建 输出 

unlink($_GET[channel name]); /删除 指定 的 .m3u 文件 
ge 


| 专家 点 评 


.m3u 文件 不 是 音频 文件 ， 它 是 音频 文件 的 列表 文件 。 打 开 它 ， 播 放 软件 并 不 会 播放 它 ， 而 


| 是 根据 它 的 记录 找到 网 络 地 址 进行 在 线 播放 。.m3u 文件 很 小 ， 因 为 它 里 面 没有 任何 音频 数 
| 据 。m3u 格式 的 文件 只 是 一 个 目录 文件 ,提供 了 一 个 指向 其 他 位 置 的 音频 视频 文件 的 索引 ， 播 
， 放 的 还 是 那些 被 指向 的 文件 ， 用 记事 本 打开 .m3u 文件 可 以 查看 所 指向 文件 的 地 址 及 文件 的 属 
| 性 ， 以 选用 合适 的 播放 器 播放 。 


问题 244 如何 将 选 定 的 音乐 添加 到 播放 列表 中 ? 


| 问题 阐述 


在 线 音乐 网 站 中 可 以 一 次 性 选择 多 个 要 播放 的 音乐 ， 并 将 其 添加 到 播放 列表 中 ， 可 以 选择 


| 循环 播放 ， 也 可 以 选择 连续 播放 。 那 么 这 个 功能 是 如 何 实现 的 呢 ? 
| 专家 解答 


播放 列表 类 似 于 电子 商务 网 站 中 的 购物 车 ， 为 用 户 选 定 的 歌曲 提供 一 个 临时 存放 的 空间 。 


实现 的 原理 是 将 用 户 选 定 歌曲 的 ID 值 存 储 到 SESSION 变量 中 ， 在 播放 列表 中 输出 SESSION 
| 变量 中 的 数据 ， 重 新 读 取 歌 曲 的 信息 ， 播 放歌 曲 。 


首先 ， 要 将 提交 的 音乐 ID 存储 到 SESSION 变量 中 ， 并 且 判 断 SESSION 中 是 否 已 经 存在 


| 该 音乐 。 其 具体 方法 如 下 ; 


<2?php 
session start(); 
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include "conn/conn.php"; 
S$u=$_SERVER[HTTP_HOST']: 
$url="http:/"."$u"; 


这 $_ POST['Submit3!] 一 "添加 到 播放 列表 "){ 
if($_POST['checkbox']—""){ 
echo "<script>alert(' 您 没有 选择 要 添加 的 音乐 ! ); history.backQ:</script>"; 
}else{ 
foreach($ POST[checkbox] as $key=>$value){ /以 book 数组 做 循环 ， 输 出 键 和 值 
if($_SESSION["music list"]=—""){ 
$_SESSION["music list"]=$value."@"; | 
jelse{ | 
$array=explode("@",$_SESSION["music list"]): | 
这 in _array($value,$array)){ | 
echo "<scripf>alert( 播 放 列表 中 已 经 存在 此 音乐 ! "):history.back():</script>"; 


exit; 
} 
$_SESSION["music list"].=$value."(@"; 
"<script>window.location.href='music list.php':</script>"; 
人 


if($_ POST['checkbox']—""){ 
echo "<script>alert( 您 没有 选择 要 播放 的 音乐 ! '); history.back();</script>"; 
jelse{ 
$date=date("His"); 


?> 


然后 ， 读 取 SESSION 变量 中 存储 的 音乐 ID， 根 据 该 ID 从 数据 库 中 查询 出 指定 音乐 文件 | 
的 存储 地 址 ， 并 生成 .m3u 文件 ， 最 终 通过 embed 标签 完成 音乐 的 循环 播放 或 连续 播放 。 其 方 | 
法 应 用 如 下 : | 
<center> 
<2php | 
这 !$fopen = fopen ("upfiles/video/$date.m3u","w")){ V 使 用 添加 模式 打开 文件 ， 文 件 指针 指 在 表 尾 | 
echo "打开 .m3u 文件 失败 ! ": | 
exit ; | 
} | 
$len=strripos($ SERVER['REQUEST_URI],"/"): | 
Stmp=substr($_SERVER[PHP_SELF'].0.Slen): /获取 文件 的 根 目录 | 
foreach($_POST['checkbox'] as $key=>$value){ ”// 以 book 数组 做 循环 ， 输 出 键 和 值 

$L sqlstr = "select id,style,name,actor,remark,address from tb_video where id='$value'order by id"; 

$l Ist = $conn->execute($]_sqlstr); 

这 Ifwrite ($fopen,"".$url.$tmp."/".$1 rst->fields[5]."){ | 
print " 写 入 内 容 失败 ! " ; | 
exit ; 

} 

if(!fwrite ($fopen,"\n\r"){ 
print " 写 入 内 容 失败 ! ": 
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exit ; 


贿 六 | <embed src="upfiles/video/<?php echo $date:?>.m3u" loop="<?php if($_POST['Submit] 一 "连续 播 
二 天 放 "){echo "false";}elseif($_POST['Submit2"]= 一 "循环 播放 "){echo "true";}?>" align="middle" ShowStatusBar 


,=true></embed> 
</center> 
| 专家 点 评 


| 在 上 述 播放 列表 中 ， 其 核心 是 SESSION 会 话 的 应 用 , 通过 SESSION 将 数据 临时 存储 于 服 
| 务 器 中 , 使 其 可 以 在 会 话 的 有 效 时 间 内 使 用 , 当 会 话 关闭 之 后 , SESSION 变量 的 值 将 自动 销毁 ， 
| 不 会 给 服务 器 的 运行 带 来 过 大 的 压力 。 


问题 24$ ”如 何 判 新 在 线 音乐 文件 的 地 址 是 否 正 确 ? 


| 问题 阐述 


| 在 判断 一 个 网 站 地 址 格式 是 否 正 确 时 ， 可 以 使 用 正则 表达 进行 判断 ， 也 可 以 根据 地 址 的 类 
， 型 和 文件 格式 来 进行 判断 。 下 面 将 探讨 如 何 判 断 在 线 音 乐 文件 的 地 址 是 否 正 确 。 


| 专家 解答 


| 判断 在 线 音 乐 文件 的 地 址 是 否 正确 ， 首 先 必须 了 解 在 线 音乐 文件 的 地 址 格式 ， 例 如 
| http://www.mrbccd.com/mp3/mrsoft.mp3, 这 就 是 一 个 名 称 为 mrosftmp3 的 音乐 文件 的 访问 地 址 ， 
| 那么 我 们 该 如 何 判断 这 个 地 址 的 格式 是 否 正确 呢 ? 
| 可 以 从 两 个 方面 入 手 ， 首 先 判 断 其 是 否 是 一 个 网 址 ， 即 判断 其 是 否 以 “http://” 开 头 ， 然 
| 后 判断 音乐 文件 的 后 级 ， 可 以 是 .mp3 或 .avi 等 多 媒体 格式 文件 。 由 此 两 处 入 手 ， 基 本 上 可 以 判 
| 断 出 文件 地 址 是 否 正确 。 其 具体 方法 如 下 : 
| 封装 JS 脚本 文件 ， 通 过 forml.intemet_address.value.substr(0，7)! ="http:/" 判 断 网 址 的 开 
| 始 位 置 是 否 符合 http://。 其 基本 用 法 如 下 : 
| if(forml.intemet_address.value.substr(0, 7)!="http://"){ 

alert(" 输 入 的 网 络 地 址 的 格式 不 正确 !); 

form!l.internet_address.select(O; 

Teturn false; 


} 


| 然后 通过 forml.internet_address.value.substr(form].internet_address.value.lastIndexOf(',")) 判 
， 断 网 址 中 文件 的 后 组 是 否 符合 指定 的 要 求 。 其 基本 用 法 如 下 : 
// 判 断 指定 字符 串 中 的 格式 符合 .mp3 
leng=form1 .internet_address.value.substr(form!1.intemet_address.value.lastIndexOf('.")); 
if(leng!=".mp3" && leng!=".rm"&&leng!=".wav"&&leng!=".mid"){ 
alert(" 输 入 的 网 络 地 址 的 格式 不 正确 1"); 
forml.intemet address.selectO: 
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Tetum false; 
| 
只 要 提交 的 音乐 文件 的 网 址 符合 上 述 两 个 要 求 ， 那 么 它 就 是 一 个 合格 的 在 线 音乐 文件 的 | 
地 址 。 | 
专家 点 评 
ee RR 
通过 上 述 方法 判断 在 线 音乐 文件 的 地 址 是 否 正确 ， 只 能 说 它 基 本 符合 个 在 线 音乐 文件 ， 


地 址 的 要 求 ， 至 于 这 个 地 址 具体 是 否 存 在 ， 只 能 通过 实践 来 检验 了 ， 而 我 们 的 判断 只 能 对 这 个 | 
地 址 起 到 一 个 规范 的 作用 。 | 


问题 246 ”如 何在 网 页 中 播放 FLV 文件 ? 


问题 阐述 
如 何在 网 页 中 播放 FLV 文件 ? 
专家 解答 | 
要 想 在 网 页 中 播放 FLV 文件 ， 首 先 应 该 在 网 页 中 其 入 Flash 播放 器 ， 当 该 播放 加入 成功 | 
后 ， 应 用 该 播放 器 播放 FLV 文件 。 
嵌入 Flash 播放 器 的 方法 有 很 多 ， 这 里 应 用 JS 嵌入 Flash 播放 器 。 具 体 实现 方法 如 下 : 
(1) 下 载 播放 器 及 其 js 文件 。 
(2) 在 HIML 页 面 头 部 <head> 区 典 入 这 个 脚本 文件 : 
<script type="text/javascript" sre="swfobject.js"></script> 
(3) 在 HTML 中 编写 一 个 用 来 放 Flash 的 容器 ， 例 如 <div>， 并 设置 其 ID， 如 flash5。 
<div id="flash5" class="right">FlvPlayer 播放 器 </div> | 
(4) 编写 js 脚本 ， 实 例 化 SWFObject 对 象 ， 载 入 FlvPlayer 播放 器 ， 并 县 设置 其 中 的 各 种 
参数 ， 指 定 要 播放 的 文件 ， 其 代码 如 下 : 


<script type="text/javascript"> 
Var s5 = new SWFObject("FlvPlayerV2009.swf","mediaplayer","500","320","8"); 
/实例 化 对 象 ， 设 置 播放 器 参数 
s5.addParam("allowfullscreen","true"): 


s5.addVariable("width","500"): // 设 置 播放 器 宽度 
s5.addVariable("height","320"); // 设 置 播放 器 高 度 
s5.addVariable("image","images/flash3.jpg"): // 载 入 背景 文件 
s5.addVariable("file"."player.flv"): // 指 定 播放 文件 
s5.addVariable("backcolor","Oxff8c00"); /背景 颜色 
s5.addVariable("frontcolor","OxE2FOFE"); /字体 颜色 
s5.write("flash5"); // 写 入 文件 
</script> 


运行 结果 如 图 13.5 所 示 。 
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图 13.5 歌词 文件 中 存储 的 内 容 


问题 247 如何 实 现 MP3 在 线 点 播 ? 


| 问题 阐述 
网 上 的 很 多 音乐 都 提供 在 线 收听 和 在 线 下 载 ， 那 么 如 何 才能 实现 MP3 的 在 线 点 播 呢 ? 
| 专家 解答 


| 在 网 页 中 嵌入 MP3 播放 器 是 实现 MP3 在 线 点 播 的 关键 , HTML 语言 对 该 功能 的 实现 提供 
| 了 良好 的 支持 , 使 用 HTML 语言 中 的 <objecP> 标 签 就 可 以 调用 系统 自 带 的 Media Player 播放 器 ， 
| 然后 使 用 <param> 标 签 传递 需 要 播放 文件 的 地 址 等 参数 ， 这 样 即 可 在 网 页 中 嵌入 能 够 播放 mp3 
| 音乐 的 播放 器 。<object> 标 签 的 基本 语法 如 下 : 
| ‘<object classid="clsid:22D6F312-BOF6-11D0-94AB-0080C74C7E95" height="68" id="MediaPlayer1" 
| width="460"> 
| <param name="AutoStart" value="-1"> 

<param name="ShowStatusBar" value="-1"> 

<param name="Filename" value="<?php echo $_GET[id]?>"> 

</object> 

参数 说 明 如 下 : 
| @ AutoStart: 该 参数 的 值 可 以 取 0 和 -1， 取 -1 表示 打开 网 页 后 自动 播放 ， 而 取 0 表示 只 有 
| 单 击 播放 器 控制 面板 中 的 播放 按钮 才 开 始 播放 。 
| @ ShowStatusBar: 该 参数 的 值 可 以 取 0 和 -1， 取 -1 表示 播放 器 有 状态 拦 ， 而 取 0 表示 播放 
| 器 无 状态 栏 。 
@ Filename: 该 参数 用 来 指定 播放 文件 的 地 址 。 
应 用 示例 


本 示例 实现 一 个 MP3 文件 在 线 点 播 的 功能 。 其 具体 步骤 如 下 : 
(1) 将 <objecte> 标 签 嵌入 到 网 页 中 ， 代 码 如 下 : 
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‘<object classid="clsid:22D6F312-BOF6-11D0-94AB-0080C74C7E95" height="68" id="MediaPlayer1" | 
width="460"> | 

<param name="AutoStart" value="-1"> 

<param name="ShowStatusBar" value="-1"> 

<param name="Filename" value="<?php echo $_ GET[id]:?>"> 
</object> 


(2) 通过 单 击 “ 试 听 ” 文 字 超 链接 实现 MP3 的 调用 与 播放 ， 代 码 如 下 : 
<a href="index.php?id=mp3/music.mp3"> 试 听 </a> 


运行 本 示例 ， 单 击 所 选 歌曲 名 称 后 面 的 “试听 ”文字 超 链 接 ， 将 弹出 如 图 13.6 所 示 的 播放 
器 ， 可 以 通过 该 播放 器 的 控制 面板 对 音乐 进行 播放 控制 。 | 


AL CopyRi thts e reserved 2011 二 林 省 mw 科技 有 限 公司 


图 13.6 MP3 在 线 点 播 
专家 点 评 
Object 标签 是 用 于 Windows IE3.0 及 以 后 浏览 器 或 者 其 他 支持 Activex 控件 的 浏览 器 。 如 果 
用 户 没 有 安装 过 flash player 那么 IE3.0 以 后 的 浏览 器 会 跳出 一 个 提示 框 访 问 是 否 要 自动 安装 | 


flash player。 可 以 通过 “classid” 和 “codebase” 属 性 来 告诉 浏览 器 自动 下 载 flash player 的 地 址 。 | 
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至 至 至 至 于 至 至 于 于 于 于 于 至 于 至 至 至 至 至 于 至 至 至 至 


问题 248 
问题 249 
问题 250 
问题 251 
问题 252 
问题 253 
问题 254 
问题 255 
问题 256 
问题 257 
问题 258 
问题 259 
问题 260 
问题 261 
问题 262 
问题 263 
问题 264 
问题 265 
问题 266 
问题 267 
问题 268 
问题 269 
问题 270 
问题 271 


如 何 安装 和 配置 Smarty 模板 ? 

如 何 将 Smarty 的 配置 方法 封装 到 类 中 ? 

如 何 通过 让 语句 判断 当前 用 户 的 权限 ? 

在 Smarty 模板 中 ， 如 何 生成 数字 验证 码 ? 

在 Smarty 模板 中 ， 如 何 直接 定义 CSS 样式 ? 

在 Smarty 模板 中 ， 如 何 谋 入 JavaScript 脚本 ? 

如 何 应 用 html option 函数 向 下 拉 列 表 中 添加 列表 项 ? 
section 语句 应 用 的 注意 事项 有 哪些 ? 

如 何 通过 section 循环 输出 数据 ? 

如 何 对 Smarty 模板 中 ， 如 何 实现 数据 进行 分 页 显示 ? 
在 Smarty 模板 中 ， 如 何 实现 日 期 、 时 间 的 格式 化 输出 ? 
在 Smarty 模板 中 ， 如 何 应 用 正则 表达 式 ? 

在 Smarty 模板 中 ， 如 何 对 关键 字 描 红 ? 

在 Smarty 模板 中 ， 控 制 输出 字符 串 的 行 宽 ? 

如 何 应 用 register object 方法 注册 模板 对 象 ? 

如 何 开启 网 站 页 面 的 缓存 ? 

如 何 判断 同一 模板 中 的 多 个 缓存 文件 ? 

如 何 通过 配置 文件 定义 变量 ? 

在 Smarty 模板 中 ， 如 何 截 取 中 文字 符 串 ? 

在 Smarty 中 如 何 进行 编码 ? 

如 何在 Smarty 中 定义 注册 模板 函数 ? 

如 何在 Smarty 中 设计 页 面 ? 

如 何在 Smarty 中 制作 日 历 ? 

如 何 应 用 Smarty+PDO 完成 数据 的 分 页 显示 ? 


需 
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问题 248 ”如 何 安装 和 配置 Smarty 模板 ? 


问题 阐述 医 网 


Smarty 模板 引擎 属于 第 三 方 的 组 件 ， 要 应 用 其 实现 网 页 的 动静 分 离 ， 首先 就 要 从 网 站 上 下 
载 该 组 件 ， 并 且 将 其 放置 到 具体 的 项 目 中 ， 在 进行 配置 后 才 可 以 使 用 。 那 么 如 何 下 载 、 安 装 和 | 
配置 Smarty 模板 呢 ? 我 们 将 在 这 里 揭晓 谜底 。 | 
专家 解答 
1， 安 装 Smarty 模板 | 
(1) 将 下 载 的 Smarty 压缩 包 解压 后 出 现 一 个 libs 目录 , 这 里 包含 了 Smarty 类 库 的 核心 文 | 
件 ， 包 括 smarty.classphp、smarty_Compilerclassphp、config File.classphp 和 debug.tpl 4 个 文 | 
件 ， 还 有 intemals 和 plug-ins 两 个 目录 ， 如 图 14.1 所 示 。 | 
(2) 复制 libs 目录 到 服务 器 根 目录 指定 的 文件 夹 下 。 至 此 ，Smarty 安装 成 功 。 
2. Smarty 模板 引擎 配置 
Smarty 模板 引擎 的 配置 步骤 如 下 : | 
(1) 确定 Smarty 类 库 的 存储 位 置 。 因 为 Smarty 类 库 是 通用 的 ， 每 一 个 项 目 都 可 能 会 使 用 | 
到 它 。 所 以 将 Smarty 存储 在 根 目录 下 是 一 个 比较 理想 的 选择 , 存储 于 根 目录 的 Smarty 文件 夹 下 。 | 
(2) 在 Smarty 目录 下 新 建 4 个 目录 templates、templates c、configs 和 cache， 如 图 14.2 | 
所 示 。 | 


文件 三 编辑 I) 查看 WY) 收藏 @) 工具 I) 帮助 00 | 
TE 


其 位 轩 = 
人 和 x 
Se 
到 
加 snsty_comple.caeephp 小 Cd 
9 先 


一 一 一 一 | 
区 后 不 开 象 


图 14.1 Smarty-2.6.23 文件 夹 图 14.2 Smarty 目录 中 的 内 容 


说 明 : ! 
新 建 的 4 个 目录 并 没有 固定 的 存储 位 置 ， 只 要 配置 的 路 径 正确 ， 存储 在 根 目录 下 的 任何 位 | 
置 都 可 以 。 | 


(3) 创建 配置 文件 。 当 所 有 需要 的 资源 都 就 绪 之 后 ， 下 面 要 做 的 就 是 将 它们 进行 整合 、 | 
调动 起 来 。 这 就 是 配置 文件 的 作用 ， 定 义 服务 器 、Smarty 的 绝对 路 径 ， 加载 Smarty 类 库 文件 ， | 
为 创建 的 4 个 目录 设置 正确 的 路 径 ， 定义 Smarty 的 定 界 符 等 操作 都 可 以 在 配置 文件 中 设置 。 | 

通常 将 配置 文件 定义 到 一 个 单独 的 文件 中 ， 在 需要 使 用 时 直接 通过 包含 语句 调用 即 可 。 这 | 
里 将 配置 文件 存储 到 config php 中 ， 其 代码 如 下 : | 


<2php 
define(BASE_ PATH',S$ SERVER[DOCUMENT ROOTT); /定义 服务 器 的 绝对 路 径 
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Er 

Ci% 
define(0SMARTY PATH'.\"MR\14 \Smarty\"): /定义 Smarty 目录 的 绝对 路 径 
require BASE PATH SMARTY PATH .'libs\Smarty.class.php': // 加 载 Smarty 类 库 文件 
S$smarty = new Smarty: /实例 化 一 个 Smarty 对 象 


$smarty->template_ dir=BASE PATH.SMARTY PATH''templates/: /定义 模板 文件 存储 位 置 
$smarty->compile_dir = BASE PATH.SMARTY PATH.'templates_c/';// 定 义 编译 文件 存储 位 置 
$smarty->config dir= BASE PATH.SMARTY PATHconfigs/: /定义 配置 文件 存储 位 置 
$smarty->cache dir= BASE PATH.SMARTY PATH.cache/: /定义 缓存 文件 存储 位 置 
/* 定义 定 界 符 寺 
$smarty->left_ delimiter = "<{'; 
S$smarty->right_delimiter = "}>"; 
9>. 

上 述 配 置 文件 的 参数 说 明 如 下 : 

@ BASE PATH: 指定 服务 器 的 绝对 路 径 。 

@ SMARTY _ PATH: 指定 Smarty 目录 的 绝对 路 径 。 

@@ require0: 加 载 Smarty 类 库 文 件 Smarty.classphp。 

@ $smarty: 实例 化 Smarty 对 象 。 

@ $smarty->template_dir: 定义 模板 目录 存储 位 置 。 

@ $smarty-> compile dir: 定义 编译 目录 存储 位 置 。 

@ $smarty-> config dir: 定义 配置 文件 存储 位 置 。 

@ $smarty-> cache_dir: 定义 模板 缓存 目录 。 

$smarty->left_delimiter: 定义 Smarty 使 用 的 开始 定 界 符 。 

@ $smarty->right delimiter: 定义 Smarty 使 用 的 结束 定 界 符 。 


关于 定 界 符 的 使 用 ， 开 发 者 可 以 指定 任意 的 格式 ， 也 可 以 不 指定 。Smarty 默认 的 定 界 符 是 
EE 和 的 

至 此 , Smarty 的 配置 讲解 完毕 。 至 于 将 配置 文件 存储 在 什么 位 置 , 可 以 根据 实际 情况 而 定 。 
专家 点 评 

Smarty 是 一 个 使 用 PHP 编写 的 应 用 于 PHP 的 模板 引擎 ， 它 将 一 个 应 用 程序 分 成 两 部 分 : 
视图 和 逻辑 控制 。 简 单 地 讲 ， 就 是 将 UI (用 户 界面 ) 和 PHP Code (PHP 代码 ) 分 离 。 这 样 ， 
程序 员 在 修改 程序 时 ， 就 不 会 影响 到 页 面 设计 ， 而 美工 在 重新 设计 或 修改 页 面 时 ， 也 不 会 影响 
程序 逻辑 。 

应 用 上 述 方法 配置 Smarty 模板 ， 当 存储 实例 的 文件 夹 发 生变 化 时 ， 就 必须 修改 config.php 
配置 文件 中 常量 SMARTY PATH 的 值 ， 否 则 将 无 法 找到 Smarty 类 库 的 存储 位 置 。 


问题 249 如何 将 Smarty 的 配置 方法 封装 到 类 中 ? 


问题 阐述 
直接 定义 Smarty 模板 的 配置 方法 , 当 存 储 实例 的 文件 夹 发 生变 化 时 , 就 必须 修改 config.php 
配置 文件 中 常量 SMARTY PATH 的 值 ， 否 则 将 无 法 找到 Smarty 类 库 的 存储 位 置 。 那 么 是 否 可 
以 将 Smarty 模板 的 配置 方法 封装 起 来 ? 这 样 无 论 实例 的 位 置 如 何 改变 ， 都 不 会 影响 到 Smarty 
. 390 . 
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将 Smarty 配置 方法 封装 到 类 中 ， 通 过 类 中 方法 实现 Smarty 模板 引擎 中 方法 的 调用 ， 这 样 | 
无 论 实例 位 置 如 何 改变 ， 都 不 会 影响 到 Smarty 的 应 用 。Smarty 配置 类 的 封装 方法 如 下 : | 

(1) 创建 system.smarty.inc.php 文件 。 首 先 ， 包 含 Smarty 类 文件 Smarty.class php。 然 后 ， 
定义 SmartyProject 类 ， 继 承 Smarty 父 类 。 最 后 ， 定 义 SmartyProject() 方 法 ， 设 置 Smarty 中 模 | 
板 文件 〈templates)、 编 译文 件 〈templates c)、 配 置 文件 〈configs) 和 缓存 文件 〈cache) 的 存 | 
储 位 置 。 配 置 文件 system.smarty.inc.php 的 代码 如 下 : 


<?php 
require("../Smarty/Smarty.class.php"); /调用 Smarty 文件 
class SmartyProject extends Smarty{ // 定 义 类 ， 继 承 Smarty 父 类 
function SmartyProjectO{ /定义 方法 ， 配 置 Smarty 模板 
S$this->template_dir = "./"; // 指 定 模 板 文件 存储 在 根 目录 下 
$this->compile_dir="./system/Smarty/templates_c/"; // 指 定编 译文 件 存 储 位 置 
S$this->config dir="./system/Smarty/configs/": 
S$this->cache_dir = "./system/Smarty/cache/": 
} 
b 
> 


(2) 既然 已 经 将 Smarty 的 配置 方法 存储 到 一 个 类 中 ， 那 么 就 需要 对 类 进行 实例 化 ， 根 据 
返回 的 对 象 名 称 调用 Smarty 中 的 方法 ， 类 的 实例 化 操作 在 system.inc.php 文件 中 完成 ， 其 返回 | 
对 象 名 为 Ssmarty。 其 代码 如 下 : 


<?php 
require("system.smarty.inc.php"); // 调 用 类 文件 
$smarty=new SmartyProject(); /执行 类 的 实例 化 操作 
> 
专家 点 评 


通过 此 方法 配置 Smarty 模板 的 好 处 是 : 无 论 将 程序 复制 到 哪个 服务 器 下 执行 ， 都 不 需要 
更 改 服务 器 或 Smarty 文件 的 绝对 路 径 ， 程 序 都 可 以 直接 运行 。 | 


问题 230 如何 通过 讶 语句 判断 当前 用 户 的 权限 ? 


问题 阐述 | 
在 动态 PHP 文件 中 可 以 通过 SESSION 变量 的 值 判断 当前 用 户 是 否 具有 访问 权限 ， 而 在 | 
Smarty 模板 页 中 不 可 以 使 用 SESSION 变量 ， 那 么 该 如 何 判断 用 户 是 否 具有 访问 权限 呢 ? 
专家 解答 | 
在 Smarty 模板 页 中 要 判断 用 户 权限 可 以 使 用 让 语句 来 实现 ， 而 关键 是 让 语句 和 模板 变量 | 
的 结合 运用 ， 在 站 语 句 中 判断 模板 变量 的 值 是 耕 为 T， 如 果 为 则 说 明 具 备 访问 权限 ;如果 模 
板 变量 值 为 F 则 说 明 不 具备 访问 权限 。 
. 391 。 Gs 
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| Smarty 模板 中 的 站 与 PHP 中 的 让 大 同 小 异 。 需 要 注意 的 一 点 是 ，Smarty 模板 中 的 让 必须 
| 以 /让 为 结束 标记 。 其 语法 格式 如 下 : 


{站 条 件 语句 1 
EN 
| {elseif 条 件 语 句 2} 
| 语句 2 
(ele) 
| 语句 3 
| if 
应 用 示例 


应 用 站 语句 在 Smarty 模板 页 中 判断 用 户 是 否 具有 访问 权限 。 其 具体 步骤 如 下 : 
| (1) 创建 system 文件 夹 ， 封 装 Smarty 模板 的 配置 方法 ， 创 建 存储 编译 文件 、 缓 存 文件 和 
| 配置 文件 的 文件 夹 。 
| (2) 创建 ndex.php 文件 ， 包 含 Smarty 配置 文件 ， 指 定 Smarty 的 模板 页 index.html。 
| (3) 创建 Smarty 模板 页 index.html， 设 计 用 户 登 录 页 面 ， 添 加 用 户 登录 的 表单 元 素 ， 并 
| 且 将 用 户 的 登录 信息 提交 到 index_ok.php 文件 。 
| (4) 创建 index_ok.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 判 断 提交 的 用 户 名 和 密码 是 否 正 
| 确 ,如 果 正 确 则 通过 display0 方 法 指定 模板 页 main html, 并 且 为 模板 变量 competence 赋值 为 T; 
| 否则 指定 到 模板 页 main html， 为 模板 变量 competence 赋值 为 F。 其 关键 代码 如 下 : 


<?php 
header ( "Content-type: text/html:; charset=UTF-8" ): /设置 文件 编码 格式 
include("system/system.inc.php"): // 包 含 配 置 文 件 


这 $_ POST['Submit]!="" && $ POST['user']!="" && $_ POST['pass']!=""){ 
if($_POST['user]—"mr" && $_ POST['pass']—"mrsoft"){ 


$smarty->assign("competence","T"): // 为 模板 变量 赋值 
S$smarty->display('main.html"): // 指 定 模 板 页 
}else{ 
$smarty->assign("competence","F"); 
$smarty->display(main html): 
}else{ 
echo "<script>alert( 用 户 名 和 密码 不 能 为 空 ! "); window.location.href='index.php';</script>"; 
} 
?> 


| (5) 创建 main.html 模板 页 。 首 先 应 用 站 语句 对 模板 变量 的 值 进行 判断 ， 如 果 值 为 了 则 输 
| 出 本 页 内 容 ， 如 果 值 为 F 则 给 出 提示 信息 。 其 关键 代码 如 下 : 
| {if $competence=—"T"} 

/省 略 了 部 分 内 容 

{请 

{1f $competence—"F"} 

<script>alert(' 您 没有 权限 访问 ， 请 重新 登录 ):windowlocation hre 全 "index.php":</scripf> 

Wi 


运行 结果 如 图 14.3 所 示 。 


了 。392 。 


图 14.3 让 语句 判断 用 户 是 否 具备 访问 权限 


专家 点 评 


Smarty 模板 计 条 件 语句 的 修饰 词 ， 除 了 可 以 使 用 PHP 中 的 <、>、=、 上 等 常见 运算 符 外 ， | 
还 可 以 使 用 eq、 ne、 neq、 gt、 lt、 lte、le、gte、ge、is even、is odd、isnot even、isnot odd、not、 | 
mod、div by、even by、odd by 等 修饰 词 修饰 。 | 


问题 231 在 Smarty 模板 中 ， 如 何 生成 数字 验证 码 ? 


问题 阐述 
在 动态 PHP 中 可 以 生成 数字 验证 码 ， 在 Smarty 模板 中 ， 如 何 生成 数字 验证 码 ? 
专家 解答 | 
在 Smarty 模板 中 可 以 应 用 foreach 语句 在 模板 页 中 直接 生成 一 个 数字 验证 码 。 严格 来 说 这 个 | 
验证 码 不 是 在 Smarty 模板 页 中 生成 的 , 只 是 通过 foreach 语句 将 PHP 动态 页 中 生成 的 验证 码 在 | 
模板 页 中 输出 。 因 此 要 实现 在 Smarty 中 生成 验证 码 的 关键 是 foreach 语句 的 应 用 和 验证 码 的 生成 。 
(1) 应 用 mt rand0 函 数 生成 验证 码 。mt rand0 函 数 的 语法 如 下 : | 
int mt rand ( [int min, int max]) | 
如 果 mt_ rand0 函 数 没 有 提供 可 选 参数 min 和 max, 则 返回 0 到 RAND _ MAX 之 间 的 伪 随 机 数 。 
(2) 应 用 explode() 函 数 将 生成 的 验证 码 写 入 数组 中 。explode() 函 数 的 语法 如 下 : | 
array explode(string separator, string string, [int limit]) 


返回 由 字符 串 组 成 的 数组 ， 每 个 元 素 都 是 string 的 一 个 子囊 ， 它 们 被 字符 串 separator 作为 | 
边界 点 分 隔 出 来 。 如 果 设 置 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 ， 而 最 后 那个 元 素 | 
将 包含 string 的 剩余 部 分 。 | 
如 果 separator 为 空 字符 串 〈"")，explode0 函 数 将 返回 false。 | 
如 果 separator 所 包含 的 值 在 string 中 找 不 到 ， 那 么 explode0 函 数 将 返回 包含 string 单个 元 | 
素 的 数组 。 | 
如 果 参 数 limit 是 负数 ， 则 返回 除了 最 后 的 -limit 个 元 素 外 的 所 有 元 素 。 
(3) 通过 assign() 方 法 将 验证 码 数组 赋 给 模板 变量 。 
(4) 应 用 foreach 语句 循环 输出 模板 变量 中 的 数组 元 素 。foreach 语法 如 下 : 


{foreach name=foreach name key=key item=item from=arr name} 
{/foreach} | 
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参数 说 明 : 

@ name: 该 循环 的 名 称 。 

@ key: 当前 元 素 的 键 值 。 

@ item: 当前 元 素 的 变量 名 。 

@ from: 该 循环 的 数组 。 

其 中 ，item 和 from 是 必要 参数 ， 不 可 省 略 。 

应 用 示例 

应 用 Smarty 模板 中 的 foreach 语句 在 模板 页 中 直接 生成 一 个 数字 验证 码 ， 它 相对 于 在 动态 


PHP 中 开发 的 验证 码 更 加 简单 、 实 用 。 其 具体 实现 步骤 如 下 : 


(1) 创建 system 文件 夹 ， 封 装 Smarty 模板 的 配置 方法 ， 这 里 不 再 袭 述 。 
(2) 创建 index.php 文件 ， 生 成 验证 码 ， 并 且 将 验证 码 赋 给 模板 变量 ， 最 后 指定 模板 页 。 


| 其 代码 如 下 : 

| <?php 
require_once("system/system.inc.php"): // 包 含 配 置 文 件 
Sarray= explode( ', mt_rand(1000,9999)); // 生 成 随机 验证 码 
$smarty->assign('title','Smarty 模板 中 生成 数字 验证 码 '); // 将 指定 数据 赋 给 模板 变量 
$smarty->assign('content', $array); // 将 数组 赋 给 模板 变量 
$smarty->display('index.html’): // 指 定 模 板 页 
?> 


| 面 ， 否 则 跳 转 到 index php 页面 。 


| <Inpnut 
| "images/Blog 03_06.jpg" 这 


(3) 创建 index.html 模板 文件 ， 设 计 用 户 登 录 页 面 ， 添 加 用 户 登录 的 表单 元 素 。 同 时 应 用 
foreach 语句 输出 模板 变量 中 传递 的 验证 码 。 其 关键 代码 如 下 : 
<form id="forml" name="forml" method="post" action="index_ok.php"> 
<input name="user" type="text" id="user" size="10" /> 
<input type="hidden" name="checks" value=" {foreach key=key item=item from=$content} 
{Sitem} {/foreach}" /> 
<input name="check" type="text" size="8" /> 
<input name="pass" type="password" id="pass" size="10" /> 
<input type="image" name="imageField" src="images/Blog 03_03.jpe" /> 
i type="image" name="imageField2" onclick="form.reset():return false;" src= 


</form> 


Smarty 模板 中 生成 的 验证 码 运行 结果 如 
图 14.4 所 示 。 

(4) 创建 index_ok.php 文件 ， 获 取 表 单 
中 提交 的 数据 ， 对 用 户 登录 信息 进行 验证 。 如 
果 用 户 名 和 密码 正确 ， 则 跳 转 到 main.php 页 


用 户 名 : Fr | 验证 码 : 9 1999 
| (5) 创建 main.php 和 main html 文件 ， Ea em RE as 
并 将 其 作为 网 站 的 主页 。 
专家 点 评 图 14.4 ”Smarty 模板 中 生成 数字 验证 码 


本 示例 中 应 用 foreach 语句 输出 验证 码 ， 如 果 要 更 新 验证 码 的 值 ， 则 可 以 单 击 刷新 按钮 或 
。394 。 


按 F5 键 ， 同 样 也 可 以 在 地 址 栏 中 重新 输入 地 址 。 


问题 232 ”在 Smarty 模板 中 ， 如 何 直接 定义 CSS 样式 ? 


问题 讲述 


由 于 在 Smarty 模板 中 默认 使 用 “{” 和 “}” 
作为 定 界 符 ， 所 以 如 果 直 接 在 Smarty 模板 中 
定义 了 CSS 样式 ， 那 么 将 输出 如 图 14.5 所 示 
的 错误 。 

因为 在 Smarty 模板 页 中 直接 将 CSS 样式 
中 使 用 的 “{” 和 “} ”解析 为 Smarty 模板 的 
定 界 符 ， 从 而 导致 出 现 错误 。 


文 作 四 
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Fatal error: Smarty emor [in index.html line 8]: syntax eror. 


in FAppScrv www MR\14Smarty Smarty.class.php on line 1092 


unrecognized tag: font-size: 12px; (Smarty_Compiler.class.php, line 446) | 
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那么 如 何 避 免 在 


Smarty 模板 中 直接 定义 CSS 样式 时 出 现 错误 呢 ? 
专家 解答 


应 用 示例 


这 里 仍然 应 用 上 一 示例 中 的 内 容 ， 唯 一 的 区 别 是 在 index.html 模板 页 中 ， 通 过 {literal} 
{/literal} 标 签 定 义 一 个 CSS 样式 ， 控 制 页 面 中 表格 、 文 字 的 背景 、 字 体 和 大 小 等 。 其 关键 代码 


如 下 : 


{literal} 
<style type="text/css"> 
<I-- 
body,td,th { 
font-size: 12px; 
} 
body { 
margin-left: Opx; 
margin-top: Opx; 
margin-right: Opx: 
margin-bottom: Opx:; 
} 
alink { 
text-decoration: none: 
} 
aivisited { 
text-decoration: none; 


a:hover { 


"395。 


图 14.5 在 Smarty 模板 中 直接 定义 CSS 样式 时 


出 现 的 错误 


为 了 避免 在 Smarty 模板 中 直接 定义 CSS 样式 时 出 现 错 误 ,Smarty 提供 了 特殊 的 处 理 方法 ， | 
即 literal 标签 ， 有 了 它 就 可 以 在 Smarty 模板 页 中 直接 定义 CSS 样式 。 

literal 标签 将 区 域内 的 数据 当 作 文本 处 理 ， 此 时 模板 将 忽略 其 内 部 的 所 有 字符 信息 。 该 特 
性 用 于 显示 有 可 能 包含 大 括号 等 字符 信息 的 CSS 样式 或 JavaScript 脚本 ， 当 这 些 信 息 处 于 
{literal} WUliteral} 标 签 中 时 ， 模 板 引擎 将 不 分 析 它 们 ， 而 直接 进行 显示 。 
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text-decoration: none; 
} 
aactive { 

text-decoration: none; 
} 
SS 
</style> 
{/literal} 


| 专家 点 评 


在 Smarty 模板 中 ， 默 认 的 定 界 符 是 “{” 和 “}” 为 了 避免 与 CSS 样式 或 JavaScript 脚本 


中 的 大 括号 冲突 ， 还 可 以 对 Smarty 模板 的 定 界 符 进行 修改 ， 使 用 的 是 $left_delimiter 和 


$right_delimiter 结束 符 变量 。 其 基本 使 用 语法 如 下 : 
$smarty->left_delimiter = "<{"; 
$smarty->right_delimiter = "}>"; 


将 定 界 符 修改 之 后 ， 就 不 会 与 CSS 样式 或 JavaScript 脚本 中 的 大 括号 冲突 了 。 
问题 253 ”在 Smarty 模板 中 ， 如 何 谈 入 JavaScript 脚本 ? 


问题 前 述 

在 Smarty 模板 中 ， 如 何 嵌入 JavaScript 脚本 呢 ? 
专家 解答 

由 于 Smarty 模板 总 是 尝试 对 “{” 和 “}” 进 行 解析 ， 所 以 想 要 直接 在 Smarty 模板 中 骨 入 
JavaScript 脚本 ， 就 必须 对 “{” 和 “}” 进 行 处 理 ， 前 面 已 经 介绍 literal 标签 对 大 括号 内 的 内 容 
进行 处 理 ， 在 这 里 将 介绍 另外 一 种 方法 来 输出 Smarty 模板 中 的 定 界 符 ， 进 而 实现 在 Smarty 模 
板 中 直接 嵌入 JavaScript 脚本 ,， 即 应 用 ldelim 和 rdelim 标签 输出 Smarty 模板 中 的 定 界 符 , 也 就 
是 说 通过 这 两 个 标签 在 Smarty 模板 中 直接 输出 “{” 和 “}” 从 而 完成 在 Smarty 模板 中 直接 霸 
入 JavaScript 脚本 的 操作 。 

应 用 示例 


在 index.html 模 板 页 中 直接 定义 JavaScript 脚 本 ,对 表单 中 提交 的 数据 进行 判断 ,通过 ldelim 
和 rdelim 标签 在 Smarty 模板 中 输出 “{” 和 “}”。 其 关键 代码 如 下 : 


<script language=javascript> 
function check form() {ldelim} 
if (form.user.value — "){ldelim} 
alert( 请 输入 用 户 名 ); 
form.user.focus(); 
Tetum false: 
{rdelim} 
if (form.pass.value — "){ldelim} 
alert( 请 输入 密码 ); 
form.pass.focus0: 
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Tetum false: 
{rdelim} 
if (form.check.value — "){ldelim} 
alert(' 请 输入 验证 码 "); 
form.check.focus(); 
Tetum false; 
{rdelim} 
{rdelim} 
</script> 


literal 标签 与 delim、rdelim 标签 的 区 别 在 于 : literal 标签 将 “{” 与 “}” 之 间 的 内 容 按 照 
普通 的 文本 文件 来 解析 ， 其 控制 的 是 内 容 ; 而 delim 与 rdelim 标签 在 Smarty 模板 中 输出 “{” 
与 “}”， 是 对 大 括号 本 身 进行 操作 。 在 Smarty 模板 中 完成 CSS 样式 或 JavaScript 脚本 的 嵌入 操 
作 时 ， 笔 者 认为 还 是 使 用 literal 标签 更 方便 一 些 。 


问题 234 如何 应用 html option 函数 向 下 拉 列 表 中 
添加 列表 项 ? 


问题 阐述 

在 开发 程序 的 过 程 中 ,都 使 用 过 下 拉 列表 框 ， 而 且 都 知道 下 拉 列 表 框 的 值 是 通过 option 标 
签 定义 的 ， 可 以 直接 在 页 面 中 定义 ， 也 可 以 使 用 从 数据 库 中 读 取 的 数据 来 定义 ， 无 论 使 用 哪 种 
方法 都 必须 应 用 option 标签 , 这 是 在 常规 的 Web 页 面 中 为 下 拉 列 表 赋值 的 方法 , 那么 在 Smarty 
模板 中 如 何 为 下 拉 列 表 框 赋值 呢 ? 
专家 解答 

在 Smarty 模板 中 可 以 通过 自 定义 函数 html options 为 下 拉 列 表 框 赋值 。 自 定义 函数 
html_options 根据 给 定 的 数据 创建 选项 组 。 该 函数 可 以 指定 哪些 元 素 被 选 定 ， 要 么 必须 指定 
values 和 ouput 属性 ， 要 么 指定 options 替代 。 自 定义 函数 语法 如 下 : 

<select name=customer id> 


{html_options values=$cust_ids selected=$customer id output=$cust_names} 
</select> 


html_options() 函 数 的 参数 说 明 如 表 14.1 所 示 。 
表 14.1 html_options() 函 数 的 参数 说 明 


参数 说 了 明 

name 下 拉 菜 单 名 称 ， 默 认 值 为 空 
options associative array 包含 值 和 显示 的 关联 数组 

selected string/array 已 选 定 的 元 素 或 元 素数 组 

output array 包含 下 拉 列 表 各 元 素 显 示 值 的 数组 
values array 包含 下 拉 列 表 各 元 素 值 的 数组 
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如 果 给 定 值 是 数组 , 则 将 作为 OPTGROUP 处 理 , 上 且 支 持 递归 。 所 有 的 输出 与 XHTML 兼容 。 
如 果 指 定 可 选 属性 name， 则 该 选项 列表 将 被 置 于 <select name="groupname"></select> 标 签 


| 中。 如 果 没有 指定 ， 那 么 只 产生 选项 列表 。 


应 用 示例 
通过 Smarty 模板 中 的 自 定义 函数 html_options 为 下 拉 列 表 框 赋值 , 更 加 适合 从 数据 库 中 读 
| 吉政 所 ， 更 符合 Smarty 开发 模式 动静 分 离 的 原则 。 其 具体 实现 步骤 如 下 : 
(1) 创建 system 文件 夹 。 首 先 ， 定 义 system.smarty.inc.php 文件 ， 封 装 Smarty 的 配置 方 


| 法 以 及 ADODB 连接 和 操作 数据 库 的 方法 。 然后 , 定义 system.inc.php 文件 ,对 Smarty 配置 类 、 
| 数据库 的 连接 和 操作 类 进行 实例 化 ， 并 返回 连接 对 象 。 最后, 创建 Smarty 文件 夹 , 定义 Smarty 


| 的 编译 文件 、 配 置 文件 和 缓存 文件 的 存储 目录 。 


(2) 创建 index.php 文件 。 首 先 ， 设 置 页 面 的 编码 格式 、 载 入 配置 文件 。 然 后 ， 生 成 随机 
验证 码 ， 并 且 将 随机 验证 码 的 值 赋 给 模板 变量 。 接 着 ， 从 数据 库 中 读 取 数 据 ， 将 读 取 的 数据 
存储 到 数组 中 ， 并 且 将 数组 赋 给 模板 变量 ， 作 为 下 拉 列 表 中 的 值 。 最 后 ， 指 定 模 板 页 。 其 代码 
如 下 : 


<2php 

header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文 件 编码 格式 
require_once("system/system.inc.php"); // 包 含 配 置 文 件 

$array= explode('', mt_ rand(1000.9999)): // 生 成 随机 验证 人 码 
$smarty->assign('title','html_option 函数 向 下 拉 列 表 中 添加 列表 项 "); // 将 指定 数据 赋 给 模板 变量 
$smarty->assign('content', $array): // 将 数组 赋 给 模板 变量 
S$res=$conn->execute("select * from tb_bccd "):; // 执 行 select 查询 语句 
$array=$res->GetArray(); /返回 查询 结果 


$array_ id=arrayO: 

Sarray_name=array(); 

for($i=0;$i<$res->RecordCount();$i++){ 
Sarray_id[]=$array[$i][id]; 
Sarray_name[]=$array[$i][name]; 


} 

$smarty->assign('cust_id', $array_id): /设置 下 拉 列 表 的 值 
$smarty->assign('cust_name', $array_name): // 设 置 列表 的 显示 数据 
$smarty->display('index.html'): // 指 定 模板 页 

es 


(3) 创建 mdex.html 模板 页 。 首 先 ， 创 建 form 表单 ， 添 加 表单 元 素 ， 将 用 户 的 注册 信息 
提交 到 index_ok.php 文件 中 ， 其 中 ， 通 过 html_options 自 定义 函数 输出 下 拉 列 表 框 的 值 ， 通 过 
foreach 语句 输出 验证 码 的 值 。 然 后 ， 应 用 literal 标签 在 页 面 中 直接 仍 入 CSS 样式 和 JavaScript 
脚本 对 用 户 提交 的 信息 进行 验证 。 其 关键 代码 如 下 : 


<form id="form" name="form" method="post" action="index ok.php" onsubmit="return check_ 
form0; "> 

词典 选择 : <select name="customer" id="customer">{html_options output=$cust_name values= $cust_id } 
</select> 

验证 码 : <input type="hidden" id="checks" name="checks" value=" {foreach key=key item=item 
from=$content} {$item} {/foreach}" /> 

<input name="check" type="text" id="check" size="8" /> {foreach key=key item=item from=$content} 
{Sitem)} {/foreach} 

<input type="image" name="imageField3" src="images/reg 07.jpg" /> 
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<input type="image” name="imageField4" onclick="form.resetO:return false:" src="images/reg_ | 
09jpg " /> | 
</form> 


生成 验证 码 和 下 拉 列 表 框 的 运行 结果 如 图 14.6 所 示 。 


图 14.6 通过 html_options() 方 法 为 下 拉 列 表 赋 值 


(4) 创建 index_ok.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 并 将 数据 添加 到 数据 表 中 。 其 代 
人 码 如 下 : | 


<?php 
header ( "Content-type: text/html: charset=UTF-8" ): /设置 文件 编码 格式 
require_once("system/system.inc.php"); // 包 含 配置 文件 


if$_POST['user']!=""&&$_POST['pass']!=""&&$_POST['checks]!=""){// 判 断 用 户 名 和 密码 是 否 为 空 
$sql="insert into tb_user(user,pass,dates,.email,type)values(".$_POST['user'].",".md5($_POST 
[pass]).", ".date("Y-m-d").",".$_POST['email].",".$_ POST['customer’].")"; 


Sres=$admindb->ExecSQL($sql. Sconn); // 执 行 insert 添加 语句 
"<script>alert( 用 户 注册 成 功 ! ); windowlocation href='main.php':</script>"; 
Le "<script>alert( 用 户 注册 失败 ! ); window.location.href='index.php';</script>"; 
jelse{ | 
echo "<scripPalert( 用户 注 册 信息 不 可 为 空 ! ); windowlocation hrefr'index.php':</script>"; | 
| 
(5) 创建 main.php 和 main.html 文件 ， 定 义 用 户 注册 成 功 后 跳 转 的 页 面 。 
专家 点 评 


在 PHP 脚本 中 , 如 果 没有 对 文件 的 编码 格式 进行 
设置 ， 那 么 在 输出 JavaScript 脚本 时 ， 如 果 输 出 的 是 
中 文字 符 串 ， 那 么 将 会 输出 乱码 ， 如 图 14.7 所 示 。 

所 以 在 PHP 中 通过 headerO 函 数 设置 页 面 的 编码 
格式 还 是 十 分 必要 的 。 其 基本 的 使 用 方法 如 下 : 

header ( "Content-type: text/html; charset=UTF-8" ): /设置 文件 编码 格式 | 
"399， \ 


图 14.7 在 JavaScript 脚本 中 输出 乱码 
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问题 255 section 语句 应 用 的 注意 事项 有 哪些 ? 


傅 。 问 题 阅 述 


section 是 Smarty 模板 中 的 数组 遍历 语句 ， 通 过 它 能 够 完成 对 模板 变量 中 传递 的 二 维 数组 
| 数据 的 循环 输出 那么 这 个 section 语句 是 如 何 应 用 的 呢 ? 在 应 用 过 程 中 需要 注意 哪些 问题 呢 ? 


| 专家 解答 


| section 用 于 遍历 数组 中 的 数据 。section 标签 必须 成 对 出 现 , 且 必 须 设置 name 和 loop 属性 。 
| 名 称 可 以 是 包含 字母 、 数 字 和 下 划 线 的 任意 组 合 。 

| 可 以 嵌 套 使 用 ， 但 必须 保证 嵌 套 的 name 唯一 。 变 量 loop (通常 是 数组 ) 决定 循环 执行 的 
| 次 数 。 当 需要 在 section 循环 内 输出 变量 时 ， 必 有 须 在 变量 后 加 上 中 括号 包含 着 的 name 变量 。 

| 输出 section 循环 内 的 变量 的 格式 是 {$arr name[sec_name]. 字 段 名 称 } 。 


问题 256 ”如 何 通过 section 循环 输出 数据 ? 


| 问题 阐述 


| 在 PHP 动态 页 中 可 以 通过 while、do...while、for 和 foreach 语句 实现 数据 的 循环 输出 ， 而 
| 在 Smarty 中 它 也 有 属于 自己 的 循环 输出 语句 foreach 和 section, 那么 如 何 通过 section 循环 输出 
| 数据 呢 ? 


| 专家 解答 
| section 循环 语句 用 于 比较 复杂 的 数组 。section 的 语法 如 下 : 


{section name=”sec_name” loop=$arr_name start=num step=num} 


参数 说 明 : 

@ name: 表示 循环 的 名 称 。 

@ loop: 表示 循环 的 数组 。 
| @@ start: 表示 循环 的 初始 位 置 ， 如 果 start=2， 那 么 说 明 循 环 是 从 loop 数组 的 第 2 个 元 素 开 
| 始 的 。 
| @ step: 表示 步 长 ， 如 果 step=2， 那 么 当 循环 一 次 后 ， 数 组 的 指针 将 向 下 移动 两 位 ， 以 此 
| 类 推 。 
| section 循环 语句 读 取 的 是 存储 在 模板 变量 中 的 数组 元 素 , 而 这 个 数组 元 素 值 是 在 动态 PHP 
| 文件 中 通过 调用 数据 库 操作 类 中 的 $admindb->ExecSQL 方法 获取 的 。 究 其 根源 就 是 应 用 
| ADODB 类 库 中 的 GetRows() 方 法 获取 的 查询 结果 。 
| 应 用 示例 

创建 会 员 浏览 模块 (vip look.php 和 vip look html)。 应 用 section 语句 循环 输出 数据 库 中 存 

的 会 员 信息 。 Ta 

(1) 编辑 vip lookphp 文件 , 定义 SQL 语句 ,调用 数据 库 操作 类 (AdminDB) 中 的 ExecSQL 


了 。400 。 
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方法 ， 查 询 出 数据 库 中 的 会 员 数 据 ， 并 且 将 返回 的 数组 赋 给 指定 的 模板 变量 。 其 代码 如 下 : 


<?php 
$sql="select * from tb_user ": // 定 义 SQL 查询 语句 
S$res=$admindb->ExecSQL($sql,$conn); // 执 行 查询 操作 
if($res){ 

$smarty->assign("res", $res); // 将 查询 结果 赋 给 指定 的 模板 变量 
} 
?> 


(2) 编辑 vip_ lookhtml 文件 ， 应 用 section 语句 循环 输 ! 


关键 代码 如 下 : 


{section name=id loop=$res} 
<tr> 
<td style="padding-bottom:5px; padding-left:5px; 
align="center" bgcolor="#FFFFFF">{$res[id].id}</td> 
<td style="padding-bottom:5px; padding-left:5px; 
align="left" bgcolor="#FFFFFF">{$res[id].user} </td> 
<td style="padding-bottom:5px:; padding-left:5px: 
align="left" bgcolor="#FFFFFF">{$res[id].email}</td> 
<td style="padding-bottom:5px: padding-left:5px: 
align="left" bgcolor="#FFFFFF">{$res[id].dates}</td> 
</tr> 
{/section} 


运行 结果 如 图 14.8 所 示 。 
明日 购物 空 昌 层 仿 管 奸 
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图 14.8 通过 section 循环 输出 数据 


专家 点 评 


8 模板 变量 中 传递 的 会 员 数 据 。 其 


padding-top:5px;" 
padding-top:5px:" 
padding-top:5px;" 


padding-top:5px; 


在 本 示例 中 应 用 section 语句 循环 输出 数组 中 的 所 有 元 素 值 ， 同 时 还 可 以 在 section 语句 中 
添加 第 3 个 参数 和 第 4 个 参数 , 即 控制 循环 的 开始 位 置 和 数组 指针 的 移动 位 置 。 示 例 代码 如 下 : 


{section name=id loop=S$res start=2 step=2} 
<tr> 

<td style="padding-bottom:Spx:; padding-left:5px; 
align="center" bgcolor="#FFFFFF">{$res[id].id}</td> 

<td style="padding-bottom:5px: padding-left:Spx: 
align="left" bgcolor="#FFFFFF">{$res[id].user}</td> 

<td style="padding-bottom:Spx: padding-left:5px; 
align="left" bgcolor="#FFFFFF">{$res[id].email}</td> 

<td style="padding-bottom:5px: padding-left:Spx: 
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| align="left" bgcolor="#FFFFFF">{$res[id].dates} </td> 
| </tr> 
{/section} 
大” 在 上 述 的 这 个 section 语句 中 ， 将 从 数组 的 第 2 个 元 素 开始 循环 ， 并 且 数 组 指针 一 次 向 下 
做 | 移动 两 位 。 还 有 就 是 在 应 用 section 语句 时 ， 切 记 一 定 要 有 /section] 结 束 标记 。 


问题 257 如 何 对 Smarty 模板 中 数据 进行 分 页 显示 ? 


| 问题 阐述 
| 在 Smarty 模板 中 如 何 对 数据 进行 分 页 显示 ? 
| 专家 解答 


| 在 Smarty 中 实现 分 页 与 在 PHP 文件 中 实现 分 页 原理 是 相同 的 , 唯一 的 区 别 是 在 Smarty 中 
| 分 页 的 处 理 操作 与 分 页 的 显示 是 分 离 的 。 如 分 页 的 处 理 操作 存储 在 sho_delete.php 文件 中 。 其 
| 应 用 到 的 关键 技术 如 下 : 
| (1) mysql_query0 函 数 ， 执 行 查 询 、 更 改 及 删除 等 操作 。 语 法 如 下 : 

Tesource mysql_query ( string query [, resource link identifier] ) 
| 参数 query 为 字符 串 类 型 ， 指 定 传 入 的 SQL 指令 ; 参数 link_identfier 为 资源 类 型 ， 指 定 传 
| 入 的 是 由 mysql_connect() 函 数 或 mysql_pconnect0 函 数 返回 的 连接 标识 ， 如 果 省 略 该 参数 ， 则 
| 会 使 用 最 后 一 个 打开 的 MySQL 数据 库 连 接 。 
| (2) mysql fetch_array0 函 数 ， 从 数组 结果 集中 获取 数据 。 语 法 如 下 : 

array mysql fetch array ( resource result [, int result_type] ) 


参数 说 明 : 

@ result: 资源 类 型 的 参数 ， 要 传 入 的 是 由 mysql_query0) 函 数 返 回 的 “数据 指针 ”。 
| @ result_type: 可 选项 , 整数 型 参数 ,要 传 入 的 是 MYSQL _ASSOC、MYSQL NUM、MYSQL_ 
| BOTH 3 种 由 PHP 定义 好 的 常数 之 一 ， 默 认 值 是 MYSQL BOTH。MYSQL ASSOC 只 得 到 关 
| 联 案 引 (相当 于 mysql fetch assoc0 函数 ); MYSQL NUM 只 得 到 数字 索引 (相当 于 
”mysql fetch_row(0) 函 数 ); MYSQL BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 。 
| (3) array0 函 数 ; 定义 一 个 数组 。 返 回 根据 参数 建立 的 数组 。 参 数 可 以 用 => 运 算 符 给 出 
| 索引 。array() 函 数 是 一 个 语言 结构 ， 用 于 字面 上 表示 数组 ， 不 是 常规 的 函数 。 语 法 如 下 : 
| array array ( [mixed...]) 


参数 mixed 的 语法 为 “key => value”， 多 个 参数 mixed 用 逗号 分 开 ， 分 别 定义 了 索引 和 值 。 
| 索引 可 以 是 字符 串 或 数字 。 如 果 省 略 了 索引 ， 会 自动 产生 从 0 开始 的 整数 索引 。 如 果 索 引 
| 是 整数 ， 则 下 一 个 产生 的 索引 将 是 目前 最 大 的 整数 索引 +1。 如 果 定 义 了 两 个 完全 一 样 的 索引 ， 
| 则 后 面 一 个 会 覆盖 前 一 个 。 
| (4) array_push0) 函 数 ， 将 数组 当成 一 个 栈 ， 并 将 传 入 的 变量 压 入 该 数组 的 末尾 。 该 数组 
| 的 长 度 将 增加 入 栈 变量 的 数目 。 返 回 数 组 新 的 单元 总 数 。 语 法 如 下 : 
| int array_push ( array array, mixed var [, mixed .了 


2 。402 。 


参数 说 明 : 
@ array: 必要 参数 。 输 入 的 数组 。 
@ var: 必要 参数 。 用 来 压 入 数组 的 值 。 
@ var2: 可 选 参数 。 用 来 压 入 数组 的 值 。 
(5) 通过 PHP 中 的 函数 、 方 法 将 从 数据 库 中 读 取 的 数据 进行 分 页 处 理 ， 通 过 $smarty 对 象 
调用 assign 方法 将 分 页 处 理 返 回 的 变量 传递 给 模板 变量 ， 并 通过 display 方法 指定 模板 页 。 
(6) 在 模板 页 sho_delete .html 中 获取 模板 变量 传递 的 值 ， 完 成 数据 的 分 页 输出 。 


应 用 示例 


本 示例 在 section 语句 循环 输出 数据 库 中 数据 的 基础 上 增加 一 个 分 页 功能 ， 实 现 对 数据 库 


中 数据 的 分 页 显示 。 其 具体 方法 如 下 : 


(1) 编辑 sho_delete.php 文件 ， 连 接 数 据 库 、 载 入 Smarty 配置 文件 ， 通 过 SQL 语句 读 取 
数据 库 中 的 数据 ， 并 且 定义 数据 的 分 页 输出 方法 ， 应 用 Smarty 中 的 assign 方法 将 分 页 输出 的 
变量 值 传递 给 模板 页 ， 最 终 指定 模板 页 。 其 代码 如 下 : 


<?php 
include_once "conn/conn.php"; 
require_once("system/system.inc.php"); 


// 执 行 查 询 语句 ， 从 数据 库 中 读 取 商 品 信 息 
$sql=mysql_query("select count(*) as totall from tb_bccd ",$conn); 


$info=mysql_fetch_aray($sql): 
S$totall=$info[totall]; 


// 连 接 数 据 库 
// 调 用 指定 的 配置 文件 


// 统 计数 据 库 中 数据 总 数 


if(empty($_GET[pages])==true || is_numeric($_GET[pages]) 一 false){ // 判 断 变 量 pages 是 否 为 空 
// 如 果 变 量 为 空 ， 则 赋值 为 1 
/如 果 不 为 室 ， 则 获取 变量 的 值 


$pagel=1; 
}else{ 
$pagel=intval($_GET[pages]); 
} 
S$pagesizel=3; 
if($totall<$pagesizel){ 
$pagecountl=1; 
jelse{ 
if($totall %$pagesizel=—0){ 


S$pagecountl=intval($totall/$pagesizel); 


jelse{ 
) 


} 
// 将 要 输出 的 数据 赋 给 assign 模板 变量 
$smarty->assign("totall", $totall); 


S$pagecountl=intval($totall/$pagesizel)+1; 


// 定 义 每 页 显示 3 条 记录 
/判断 如 果 数据 库 中 数据 小 于 每 页 显示 的 记录 数 
// 则 定义 pagecount 变量 的 值 为 1 


$smarty->assign("pagesizel",$pagesizel); 


$smarty->assign("pagel",$pagel); 


$smarty->assign("pagecountl",$pagecount]); 


$query=mysql_ query("select * from tb bccd order by id desc limit ".($pagel-1)*$pagesizel.", 


$pagesizel",$conn ); 
$myrow=mysql fetch array($query); 
S$array=array(); 
do{ 
array_push($array, $myrow); 


}while($myrow=mysql fetch array($query)); 


. 403 . 


/用 总 的 记录 数 除 以 每 页 显示 的 记录 数 ， 
获取 共有 儿 页 


// 定 义 一 个 空 数组 


// 将 获取 的 数组 值 写 入 到 新 的 数组 中 
/循环 读 取 数据 库 中 的 数据 
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j CD 
不 论 
| 这 !$array){ 
$smarty->assign("iscommo","F");// 判 断 如 果 执 行 失败 则 输出 模板 变量 iscommo 的 值 为 了 
| j}else{ 
! $smarty->assign("iscommo","T");// 判 断 如 果 执 行 成 功 , 则 输出 模板 变量 iscommo 的 值 为 了 
| $smarty->assign("arr",$array); /定义 模板 变量 arr， 输 出 数据 库 中 的 数据 
| } 


$smarty->assign(title," 商 品 删 除 "); 
?> 


| (2) 编辑 sho_delete.html 模板 页 ， 应 用 section 语句 和 模板 变量 完成 数据 的 分 页 输出 。 其 
| 关键 代码 如 下 : 


| <table width="545" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor= 
| "#3399CC"> 
| <tr> 
<td height="25" align="center" bgcolor="#FFFFFF">ID</td> 
<td align="center" bgcolor="#FFFFFF"> 商 品名 称 </td> 
<td align="center" bgcolor="#FFFFFF"> 操 作 </td> 
</tr> 
{section name=id loop=$arr} 
<tr> 
<td align="center" bgcolor="#FFFFFF">{$arr[id].id} </td> 
<td bgcolor="#FFFFFF">{$ar[id].name}</td> 
<td align="center" bgcolor="#FFFFFF"><a href="delete.php?conn_id={$arrfid].id}"> 删 除 </a></td> 
</tr> 
{/section} 
</table> 
<table width="545" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td> 图 书 &nbsp;{$totall}&nbsp; 本 &nbsp; 每 页 {$pagesizel} 本 &nbsp; 第 &nbsp; {$pagel1}&nbsp; 
页 / 共 &nbsp:;{$pagecount1}&nbsp; 页 </td> 
<td align="center">{if $pagel 一 1 } 首 页 &nbsp; 上 一 页 {else}<a href="index.php?caption= 
{$titlelescape:"url"} &pages=1"> 首 页 </a>&nbsp; 
<a href="index.php?caption={$titlelescape:"url"} &pages={$pagel-1}" > 上 一 页 </a>{/if} {if $pagel 
一 $pagecountl } 下 一 页 &nbsp; 尾 页 {else} 
<a href="index.php?caption={$titlelescape:"url"} &pages={$pagel+1}"> 下 一 页 </a>&nbsp; 
<a href="index.php?caption={ $titlelescape:"url"} &pages={$pagecount1}"> 尾 页 </a>{/if} 


</td> 
</tr> 
! </table> 
运行 结果 如 图 14.9 所 示 。 
帝 管 理 员 : 
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图 14.9 Smarty 模板 中 数据 的 分 页 输出 
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问题 258 ”在 Smarty 模板 中 ， 如 何 实现 日 期 、 
时 间 的 格式 化 输出 ? 


问题 阐述 


在 PHP 脚本 中 日 期 、 时 间 的 格式 化 输出 最 常用 的 就 是 date0 函 数 ， 那 么 在 Smarty 模板 中 ， 
如 何 完成 日 期 、 时 间 的 输出 呢 ? 


专家 解答 


在 Smarty 模板 中 应 用 date_format 函数 ， 格 式 化 strftime() 函 数 获 得 的 时 间 和 日 期 。 2 有 | 
示例 如 下 : 


{$smarty.nowldate_format:"%A, %B %e, WY":$times} 
参数 说 明 : | 
@ $smarty.now: 传递 给 date_format 的 数据 。 这 个 参数 可 以 是 在 PHP 动态 页 中 定义 的 日 期 、| 
时 间 横 板 变量 ， 也 可 以 使 用 Smarty 中 的 保留 变量 $smarty.now 或 Smarty 模板 中 的 日 期 。 
© %A， %6B %e, %Y: date_format 函数 执行 格式 化 操作 时 使 用 的 格式 。 | 
日 Stimes: 当 传递 给 date format 的 数据 为 空 时 ， 通 过 $times 设置 date_ format 格式 化 的 冉 | 
认 值 。 
date format 函数 可 以 使 用 的 转换 格式 有 很 多 ， 其 常用 的 转换 格式 说 明 如 下 : 
%a - 根据 当地 格式 输出 “星期 ”缩写 格式 
%A - 根据 当地 格式 输出 “星期 ”全 称 格式 
%b - 根据 当地 格式 输出 “月 ”缩写 格式 


%B - 根据 当地 格式 输出 “月 ”全 称 格式 
%Y- 根据 当地 格式 输出 “年 ”全 称 格式 


要 了 解 date_format 函数 所 有 可 以 使 用 的 转换 格式 , 建议 读者 参考 Smarty 模板 的 参考 手册 。 | 
应 用 示例 
在 index.html 模板 文件 中 插入 date_format 函数 ， 输 出 系统 的 当前 时 间 。 其 关键 代码 如 下 : 
<td width="480" height="18" align="left"> 当 前 时 间 : {$smartynowldate format:" %A %B - | 
%e - WY"}</td> | 
<!--{$smarty.nowldate_format} 
{$smarty.nowldate_ format:"%A, %B %e, WY":$times} 
{$smarty.nowldate_format:"%H:%M:%S"} 
{$yesterdayldate_format} 


{$yesterdayldate_format:"%A, %B %e, WY"} 
> 


专家 点 评 


通过 date_format 函数 不 但 可 以 输出 系统 的 当前 时 间 ， 如 果 配 合 strtotimeO 函 数 ， 还 可 以 获 | 
取 其 他 的 时 间 。 例 如 ， 获 取 系统 前 一 天 的 时 间 ， 其 方法 如 下 : 


首先 在 index.php 中 , 应 用 strtotimeO 函 数 获 取 前 一 天 的 时 间 戳 , 并 且 将 时 间 戳 赋 给 模板 变量 ， 
. 405 . 
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区 
te 
然后 在 index.html 模板 页 中 ， 应 用 date_format 函数 格式 化 输出 前 一 天 的 时 间 。 其 关键 代码 如 下 : 
//Index.php 文件 
$smarty->assign(yesterday', strtotime('-1 day)); 
/mdex html 文件 


{$ yesterday |date format:" %A %B -%e-%Y"} 


问题 259 ”在 Smarty 模板 中 ， 如 何 应 用 正则 表达 式 ? 


问题 阐述 
在 Smarty 模板 中 ， 如 何 应 用 正则 表达 式 ? 
解答 


在 Smarty 模板 中 通过 regex_replace 方法 调用 正则 表达 式 完 成 字符 串 的 匹配 操作 。regex_ 
replace 可 对 指定 的 字符 串 进 行 匹配 ， 其 包括 两 个 参数 : 第 一 个 是 指定 的 正则 表达 式 ; 第 二 个 是 
指定 的 替换 文本 。 其 应 用 示例 如 下 : 


{$namelregex_replace:"/[mr]/":"<font color='#FF0000> 明 日 科技 </font>"} 


在 这 个 示例 中 ,将 模板 变量 $name 中 的 “mr ” 蔡 换 为 "<font color=#FF0000> 明 日 科技 </font> ”。 


应 用 示例 
在 会 员 浏览 模块 中 ， 应 用 regex_replace 变量 将 会 员 名 称 中 的 “mr” 蔡 换 为 “明日 科技 ” 
其 具体 步骤 如 下 : 


(1) 编辑 vip_ lookphp 文件 , 分 页 读 取 数据 库 中 存储 的 会 员 数 据 ， 并 且 将 分 页 读 取 返回 的 


变量 赋 给 模板 变量 ， 最 终 指定 模板 页 。 其 关键 代码 如 下 : 


<2php 
include_once "conn/conn php": // 连 接 数据 库 
require_once("system/system.inc.php"); // 调 用 指定 的 文件 


/执行 查询 语句 ， 从 数据 库 中 读 取 商 品 信息 
$sql=mysql_query("select count(*) as totall from tb_name ",$conn); 
S$info=mysql fetch_array($sq]); 


$totall=$info[totall]: /统计 数据 库 中 数据 总 数 
这 empty($_GET[pages]) 一 tmue || is_numeric($_GET[pages]) 一 false){ /判断 变量 pages 是 否 为 空 
$pagel=1; // 如 果 变 量 为 空 ， 则 赋值 为 1 
}else{ /如 果 不 为 空 ， 则 获取 变量 的 值 
S$pagel=intval($_GET[pages)); 
} 
$pagesizel=3:; // 定 义 每 页 显示 3 条 记录 
if($totall<$pagesizel){ /判断 如 果 数据 库 中 数据 小 于 每 页 显示 的 记录 数 
$pagecountl=1; // 则 定义 pagecount 变量 的 值 为 1 
}else{ 
if($totall%$pagesizel—0){ 
S$pagecount1=intval($totall/$pagesizel); 。 // 用 总 的 记录 数 除 以 每 页 显示 的 记录 数 ， 
获取 共有 几 页 
jelse{ 


$pagecountl=intval($totall/$pagesizel)+1; 
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} 


} 
// 将 要 输出 的 数据 赋 给 assign 模板 变量 | 
$smarty->assign("totall",$totall): | 
$smarty->assign("pagesizel",$pagesizel); | 
$smarty->assign("pagel",$pagel); | 色狼 
$smarty->assign("pagecountl1",$pagecount]); 1 _ 
$query=mysql query("select * from tb name order by id desc limit ".($pagel-1)*$pagesizel.", 
$pagesizel",$conn ); | 
$myrow=mysql fetch array($query); | 


Sarray=array(); // 定 义 一 个 空 数组 | 
dof | 
array_push($array, $myrow); // 将 获取 的 数组 值 写 入 新 的 数组 中 | 
}while($myrow=mysql fetch_array($query)): // 循 环 读 取 数 据 库 中 的 数据 
if(!$array){ 
$smarty->assign("iscommo","F");// 判 断 如 果 执 行 失败 则 输出 模板 变量 iscommo 的 值 为 F 
}else{ 
$smarty->assign("iscommo","T");// 判 断 如 果 执 行 成 功 , 则 输出 模板 变量 iscommo 的 值 为 了 
$smarty->assign("arr",$array); /定义 模板 变量 arraybbstell， 输 出 数据 库 中 的 数据 
} 


$smarty->assign(title," 会 员 浏 览 "); 
人 


(2) 编辑 vip_look.html 模板 页 ， 通 过 section 语句 完成 会 员 信息 的 分 页 输出 ， 并 且 应 用 
regex_replace 变量 将 会 员 名 称 中 的 “mr” 替 换 为 “明日 科技 ”。 其 关键 代码 如 下 : 


{section name=id loop=$arr} 
<tr> 
<td align="center" bgcolor="#FFFFFF">{$arr[id].id}</td> 
<td align="left" bgcolor="#FFFFFF">{$amlid].namelregex._replace:"/[mr]/":"<font color=#FF0000> 
明日 科技 </font>"}</td> 
<td align="left" bgcolor="#FFFFFF">{$arr[id].email}</td> 
<td align="left" bgcolor="#FFFFFF">{$arr[id].dates}</td> 


</tr> 
{/section} 
运行 结果 如 图 14.10 所 示 。 
明日 购物 空 向 屋 仿 填 下 ee 
图 14.10 Smarty 模板 中 正则 表达 式 的 应 用 
专家 点 评 


在 Smarty 模板 中 不 但 通过 regex_replace 可 以 实现 字符 串 的 替换 操作 ， 而 且 使 用 replace 函 | 
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数 也 可 以 实现 字符 品 的 蔡 换 操作 。 
问题 260 在 Smarty 模板 中 ， 如 何 对 关键 字 描 红 ? 


问题 阐述 

在 动态 PHP 中 ， 可 以 通过 函数 str_replace 实现 对 关键 字 描 红 ， 在 Smarty 模板 中 ， 如 何 对 
关键 字 描 红 呢 ? 
专家 解答 

(1) Smarty 模板 中 的 关键 字 描 红 应 用 的 是 replace 变量 ， 其 实现 字符 串 的 简单 查询 和 替换 
操作 。 其 包括 两 个 参数 : 第 一 个 参数 是 被 替换 的 文本 字符 串 ; 第 二 个 参数 是 用 来 替换 的 文本 字 
符 串 。 其 应 用 示例 如 下 : 
{$articleTitlelreplace:"Garden":"Vineyard"} 


在 这 个 示例 中 ， 将 模板 变量 $articleTitle 中 的 Garden 替换 为 Vineyard。 

(2) 查询 关键 字 描 红 的 原理 : 在 PHP 动态 页 中 获取 form 表单 提交 的 查询 关键 字 , 并 且 执 
行 模糊 查询 ， 将 查询 结果 赋 给 模板 变量 ， 同 时 将 查询 的 关键 字 也 赋 给 模板 变量 ， 然 后 在 模板 页 
中 通过 section 语句 循环 输出 查询 结果 , 并 应 用 replace 将 查询 结果 中 关键 字 的 字体 设置 为 红色 。 

应 用 示例 
在 后 台 管 理 的 订单 查询 模块 中 对 查询 结果 中 的 关键 字 进 行 描 红 。 其 具体 步 又 如 下 : 

(1) 编辑 for_select.html 模板 页 ， 创 建 form 表单 ， 提 交 查 询 的 关键 字 ， 将 数据 提交 到 
for_select.php 页 面 进行 处 理 ， 同 时 在 本 页 中 应 用 section 语句 循环 输出 查询 结果 ， 并 且 应 用 
replace 对 查询 的 关键 字 进 行 描 红 操作 。 其 关键 代码 如 下 : 

{section name=id loop=$res} 
<tr> 

<td align="center" bgcolor="#FFFFFF">{$res[id].id}</td> 

<td bgcolor="#FFFFFF"> {Sres[id] numberlreplace:$select:"<font color=#FF0000'>$select</font>"} 
</td> 

<td bgcolor="#FFFFFF">{S$res[id]userlreplace:" 明 日 科技 ":"<font color=#FF0000> 明 日 科技 

</font>"}</td> 

<td bgcolor="#FFFFFF">{$res[id].dates}</td> 

<td align="center" bgcolor="#FFFFFF">{if $res[id].type 一 0} 示 处理 {else} 已 处 理 {/if}</td> 
</tr> 
{/section} 


(2) 编辑 for_select.php 动态 页 ， 以 form 表单 中 提交 的 关键 字 为 条 件 执行 模糊 查询 ， 如 果 
查询 结果 为 真 ， 则 将 查询 结果 赋 给 模板 变量 ， 同 时 将 查询 的 关键 字 也 赋 给 指定 的 模板 变量 ;如 
果 查 询 结果 为 假 ， 则 将 为 模板 变量 赋值 F， 其 代码 如 下 : 


<?php 

$smarty->assign(title,"Smarty 模板 中 的 关键 字 描 红 技 术 "); /定义 标题 变量 

if($ POST['number']!=""){ /判断 查 询 关键 字 是 否 为 空 
$sql="select * fomtb order where number like %"$ POST[mnumber]"%": /定义 模糊 查询 的 语句 
Sres=$admindb->ExecSQL($sql,$conn): // 执 行 模糊 查询 
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if($res){ 


$smarty->assign("select".$ POST['number']); // 将 查询 的 关键 字 赋 给 模板 变量 
Ssmarty->assign("boo","T"); /将 模板 变量 赋值 为 了 
Ssmarty->assign("res",$res); // 将 查询 结果 赋 给 模板 变量 

jelse{ 
$smarty->assign("boo","F"): 

} 

}else{ 
$smarty->assign("boo","F"); 


?> 


运行 结果 如 图 14.11 所 示 。 
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图 14.11 Smarty 模板 中 的 关键 字 描 红 
专家 点 评 


通过 replace 变量 不 但 可 以 实现 查询 关键 字 描 红 的 功能 ， 而 且 可 以 实现 对 论坛 中 的 非法 关 
键 字 进 行 替 换 的 功能 ， 其 实现 方法 与 关键 字 的 描 红 相 同 。 | 


问题 261 在 Smarty 模板 中 ,如 何 控制 输出 字符 串 的 行 宽 ? 


问题 阐述 


在 Web 页 面 中 通常 是 应 用 CSS 样式 来 控制 字符 串 输出 的 行 宽 ， 那 么 在 Smarty 模板 中 ， 如 | 
何 控制 输出 字符 串 的 行 宽 呢 ? | 
专家 解答 

在 Smarty 模板 中 有 一 个 变量 可 以 控制 输出 字符 串 的 行 宽 ， 它 就 是 wordwrap。 

wordwrap 指定 段落 的 宽度 〔 即 控制 一 行 显示 多 少 个 字符 ， 如 果 超 过 这 个 字符 数 则 换行 )， 
其 默认 值 是 80 字符 。 该 方法 包含 3 个 参数 : 

@ 第 一 个 参数 设置 每 行 显示 的 字符 数 。 

@ 第 二 个 参数 设置 在 约束 点 使 用 什么 字符 (默认 值 是 换行 符 \n)。 | 

日 第 三 个 参数 控制 Smarty 是 截取 到 字符 串 的 末尾 ， 还 是 精确 到 指定 长 度 的 字符 ， 默 认 情 | 
况 是 截取 到 字符 串 的 默认 值 。 如 果 设 置 该 参数 值 为 tue， 那 么 就 精确 截取 到 指定 长 度 的 字符 。 
其 应 用 的 示例 如 下 : 

{SarticleTitlelwordwrap:30} 
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设置 段落 一 行 中 包括 30 个 字符 ， 并 使 用 \n 进行 换行 操作 。 
{$articleTitlelwordwrap:30:"<br>\n"} 
| 设置 段落 一 行 中 包括 30 个 字符 ， 并 使 用 <br>n 进行 换行 操作 。 
食 和 | | {$articleTitlelwordwrap:30:"\n":true} 
设置 段落 一 行 中 包括 30 个 字符 ， 并 使 用 \n 进行 换行 操作 ， 同 时 精确 截取 30 个 字符 。 
| 上 述 的 示例 是 对 模板 变量 $articleTitle 中 存储 的 数据 进行 操作 。 
应 用 示例 
在 本 示例 中 应 用 wordwrap 变量 控制 页 面 中 输出 字符 串 的 行 宽 。 其 具体 步骤 如 下 : 
| (1) 创建 system 文件 夹 。 首 先 ， 定 义 system.smarty.inc.php 文件 ， 封 装 Smarty 的 配置 方 
| 法 。 然 后 ， 定 义 system.inc.php 文件 ， 对 Smarty 配置 类 进行 实例 化 ， 并 返回 连接 对 象 。 最 后 ， 
| 创建 Smarty 文件 夹 ， 定 义 Smarty 的 编译 文件 、 配 置 文件 和 缓存 文件 的 存储 目录 。 
| (2) 创建 index.php 文件 。 首 先 ， 设 置 页 面 的 编码 格式 、 载 入 配置 文件 。 然 后 ， 通 过 
| file_get_contents() 函 数 读 取 存 储 在 文本 文件 content txt 中 的 字符 串 ， 并 应 用 iconv0 函 数 将 字符 
| 串 有 gb2312 编码 转换 为 UTF-8 编码 ， 同 时 将 转换 后 的 文件 存储 到 模板 变量 中 ， 最 终 指定 模板 
| 页。 其 代码 如 下 : 


<2php 

header ( "Content-type: text/html:; charset=UTF-8" ): // 设 置 文件 编码 格式 
include("system/system.inc.php"): // 包 含 配置 文件 
$smarty->assign('title','Smarty 模板 中 控制 输出 字符 串 的 行 宽 "); // 定 义 实例 标题 变量 

S$str = file_get_contents('files/content.txt'); // 读 取 文 本 文件 中 的 数据 
$smarty->assign('content',iconv("gb2312","utf-8",$str)); 。 // 将 文本 文件 中 的 数据 存储 到 模板 变量 中 
$smarty->display('index.html'); // 指 定 模板 页 

?> 


| 3) 创建 indexhtml 模板 页 ， 通 过 模板 变量 输出 文本 文件 中 的 字符 串 ， 并 且 应 用 wordwrap 
| 控制 每 行 显示 30 个 字符 ， 应 用 <br>\n 完成 换行 操作 。 其 关键 代码 如 下 : 
| {$contentlwordwrap:30:"<br>\n"} 

运行 结果 如 图 14.12 所 示 。 
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专家 点 评 


在 应 用 wordwrap 控制 段落 中 每 行 显示 的 字符 时 ， 默 认 使 用 wn 为 换行 符 ， 但 当 使 用 m 作为 | 
换行 符 时 ， 对 页 面 中 字符 串 的 输出 不 会 受到 影响 ， 即 看 不 到 字符 串 换行 的 效果 ， 只 有 在 查看 源 | 
文件 时 才能 看 出 字符 串 已 经 换行 ， 所 以 ， 在 应 用 wordwrap 控制 段落 中 每 行 显示 的 字符 时 ， 应 | 
该 设置 第 二 个 参数 , 指定 <br> 为 换行 符 , 这 样 在 页 面 中 输出 时 就 可 以 直接 看 到 段落 换行 的 效果 。 ， 

另外 ， 在 Smarty 模板 中 还 可 以 通过 nl2br 将 所 有 的 换行 符 转 换 成 </br>， 其 应 用 与 PHP 中 | 


的 nl2br0 函 数 相同 。 而 本 实例 中 的 字符 串 换 行 操作 也 可 以 使 用 如 下 方法 来 完成 : 
{$contentlwordwrap:30|nl2br} 


问题 262 ”如 何 应 用 register object 方法 注册 模板 对 象 ? 


问题 阐述 


在 PHP 中 ， 可 以 通过 function 创建 自 定义 函数 ， 完 成 一 些 特殊 的 操作 ， 那 么 在 Smarty 中 


如 何 通 过 register_object 方法 注册 对 象 ， 使 之 在 模板 页 中 完成 一 些 特殊 的 操作 呢 ? 
专家 解答 
使 用 register_object 方法 注册 一 个 在 模板 中 使 用 的 对 象 ， 语 法 如 下 : 


Void register_object (string object_name，object $object, array allowed methods/properties, boolean | 


format, array block methods) 
register_object 注册 对 象 的 参数 说 明 如 表 14.2 所 示 。 
表 14.2 register_object 注册 对 象 的 参数 说 明 


参数 说 了 明 
object_name 模板 页 中 使 用 的 对 象 名 称 
$object 实例 化 的 对 象 名 称 
allowed methods/properties | 定义 允许 使 用 的 方法 ， 如 果 设 置 为 nll， 表 示 所 有 方法 都 可 以 
format 设置 对 象 参数 传递 的 方法 ， 如 果 值 为 flse， 则 表示 一 次 传递 一 个 参数 
block methods 该 参数 在 format 的 值 为 tue 应 用 ， 向 对 象 中 添加 一 个 块 方法 
应 用 示例 


在 本 示例 中 通过 register_object 方法 注册 对 象 ， 定 义 两 个 方法 moneyFormat0 和 unHtml(0)， 


分 别 对 留言 本 中 的 评分 进行 转 义 ， 保 留 两 位 小 数 ， 对 留言 内 容 中 的 html 标记 进行 转 义 。 其 具 


体 步 骤 如 下 : 


(1) 创建 system 文件 夹 ， 定 义 Smarty 文件 夹 ， 存 储 编译 目录 、 缓 存 目录 ; 创建 类 文件 | 
system.smarty.inc.php， 定 义 数 据 库 连接 、 操 作 、 分 页 和 Smarty 的 配置 类 ， 定 义 类 的 实例 化 文件 | 


system.inc.php， 完 成 各 个 类 的 实例 化 操作 ， 并 返回 操作 对 和 象 。 


(2) 创建 index.php 文件 ， 包 含 类 的 实例 化 操作 文件 ， 创 建 类 ， 定 义 评分 转 义 和 html 标记 | 
转 义 的 方法 ， 通 过 register_object 注册 模板 对 象 ; 完成 数据 库 中 数据 的 分 页 读 取 ， 最 终 指定 模 | 


板 页 。 其 代码 如 下 : 
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<?php 
Tequire_once("systemy/system_inc php"); /调用 指定 的 文件 
class Util { 
function moneyFormat($integraD) { // 将 数字 转换 成 有 两 位 小 数 的 形式 
Teturn str_replace (',', ", number format ( $integral, 2 ) ); 
b 
function unHtml($text) { // 转 换 字符 串 中 的 html 标签 
$str = htmlspecialchars ( $text ); 
$str = ereg_replace ( '<br>', \n', $str ); 
S$str = nl2br ( $str ); 
$str = ereg_replace ('', '&nbsp:", $str ); 
Teturn $str; 
: 上 
Sntil=new UtilO; // 实 例 化 对 象 


$smarty->register_object("Util", $ntil,null,false); /注册 模板 对 象 
S$array=$seppage->ShowDate("select*from tb _guestbook order by id desc",$conn,2,$ GET["page"]); 
/调用 分 页 类 ， 实 现 分 页 功能 
if(!$array){ 
$smarty->assign("iscommo","F");// 判 断 如 果 执 行 失败 ， 则 输出 模板 变量 iscommo 的 值 为 F 
Jelse{ 
$smarty->assign("iscommo","T");// 判 断 如 果 执行 成 功 , 则 输出 模板 变量 iscommo 的 值 为 
$smarty->assign("showpage",$seppage->ShowPage(" 留 言 "," 条 ","","al")); 


// 定 义 输出 分 页 数据 的 模板 变量 showpage 
$smarty->assign("array", $array):; 
} 
$smarty->assign('"title','register_object0 方 法 注册 模板 对 象 "); 
$smarty->display('index.html"); 
ee 


(3) 创建 index.html 模板 页 ,分 页 输出 模板 变量 中 传递 的 数据 ， 并 且 调用 注册 模板 对 象 中 


的 方法 对 评分 和 HIML 标记 进行 转换 操作 。 同 时 创建 一 个 form 表单 ， 用 于 提交 留言 信息 ， 将 
数据 提交 到 index_ok.php 文件 中 。 其 关键 代码 可 参考 关键 技术 中 的 内 容 。 


如 下 : 


(4) 创建 index_ok.php 文件 ， 将 表单 中 提交 的 留言 信息 存储 到 指定 的 数据 表 中 。 其 代码 


<2php 

header ( "Content-type: text/html: charset=UTF-8" ): // 设 置 文件 编码 格式 
require_once("system/system.inc.php"): // 调 用 指定 的 文件 
$title=$_ POST[title]; // 获 取 标 题 
$content=$_ POST[content]: // 获 取 内 容 
$integral=$_ POST[integral]: 

$createtime=date("Y-m-j H:i:s"): /定义 时 间 


$arr=$admindb->ExecSQL("insert into tb guestbook(title,content,createtime,integral) values 


(‘$title','$content','$createtime','$integral’)", $conn); 


这 $arnj{ 
echo "<script>alert(' 留 言 发 表 成 功 !"):window.location.href~'index.php';</script>"; 
}else{ 
echo "<script>alert(' 留 言 发 表 失 败 !"):history.back(:</script>"; 
} 


人 
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运行 结果 如 图 14.13 所 示 。 
FANXING ez NET /Nk /RR 


硬 关 图 片 立 首 志 
TPT 


更 名 联系 方式 
MT 


更 盘 电子 图 书 
Pw TSI 
共有 旱 言 ?条 每 页 显示 条 第 首页 上 一 页 下 一 页 尾 页 wa 
主题: 3 分 : 


提交 ] [重生 


图 14.13 ”Smarty 模板 中 日 期 的 格式 化 输出 


虽然 register_object 方法 注册 的 对 象 在 模板 页 中 完成 对 数据 的 操作 ， 但 是 其 具体 功能 的 实 | 

现 仍然 是 在 PHP 动态 文件 中 。 下 面 应 用 str_replace0 和 number format(O 函 数 实现 字符 串 的 替换 | 

和 数字 的 格式 化 。 | 
(1) str replace(O 函 数 ， 实 现 字 符 串 的 替换 。 语 法 如 下 : 

mixed str_replace ( mixed search, mixed replace, mixed subject , int &count ) | 

str replace() 函 数 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ,参数 &count 表 | 

示 蔡 换 字符 串 执行 的 次 数 。str_replace0) 函 数 的 参数 说 明 如 表 14.3 所 示 。 | 


表 14.3 str_replace() 函 数 的 参数 说 明 


指定 将 要 被 替换 的 字符 


指定 替换 所 使 用 的 字符 
指定 被 操作 的 字符 串 
替换 字符 串 执行 的 次 数 


(2) number format(O) 函 数 ， 将 数字 字符 串 格式 化 。 语 法 如 下 : 

string number_format ( float number , int decimals) 
string number format ( float number, int decimals, string dec_point, string thousands_sep) | 
number formatO 函 数 返回 参数 number 格式 化 后 的 字符 串 ， 该 函数 可 以 有 1 个 、2 个 或 4 个 | 
参数 ， 但 不 能 是 3 个 参数 。 | 
如 果 只 有 一 个 参数 number， 则 number 格式 化 后 会 舍 去 小 数 点 后 的 值 ， 且 每 一 干 就 会 以 逗 | 
号 〈,) 来 隔 开 。 | 
如 果 有 两 个 参数 , 则 number 格式 化 后 会 到 小 数 点 第 decimals 位 , 且 每 一 千 就 会 以 逗号 (,，) | 
来 隔 开 。 | 
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如 果 有 4 个 参数 ， 则 number 格式 化 后 会 到 小 数 点 第 decimals 位 ，dec_point 用 来 蔡 代 小 数 
| 点 (.)，thousands_sep 用 来 替代 每 一 千 隔 开 的 逗号 (, )。 


全 问题 263 ”如 何 开启 网 站 页 面 的 经 存 ? 
问题 阐述 


| 缓存 的 合理 应 用 应 该 以 实际 的 开发 需要 为 依据 ， 对 于 那些 不 需要 经 常 更 新 的 程序 开启 组 
| 存 ， 可 以 提高 网 站 的 访问 效率 ， 那 么 在 Smarty 模板 中 是 如 何 开启 网 站 缓存 的 呢 ? 
| 专家 解答 
缓存 的 设置 既 要 考虑 网 站 访问 速度 的 提高 ， 又 要 考虑 缓存 生命 周期 的 合理 性 。 不 能 偏执 一 
| 方 ， 应 该 采取 中 庸 之 道 ， 达 到 一 个 最 理想 的 效果 。 
| 例如 ， 网 站 首页 是 不 需要 经 常 更 新 的 内 容 ， 缓 存 的 生存 周期 就 可 以 设置 的 长 一 些 ， 而 类 似 
| 于 论坛 中 帖子 的 数据 ， 则 可 以 不 开启 缓存 ， 因 为 帖子 的 数据 会 不 断 地 更 新 ， 就 没有 必要 使 用 组 
| 存 ， 如 果 使 用 缓存 可 能 会 导致 浏览 不 到 最 新 的 数据 。 
| 1. 开启 缓存 
| 开启 缓存 的 方法 非常 简单 ， 只 要 将 Smarty 对 象 中 $config 的 值 设置 为 True 即 可 ,同时 还 要 
| 通过 Smarty 对 象 中 的 Scache dir 属性 指定 缓存 文件 的 存储 位 置 。 其 操作 代码 如 下 : 
| $smarty->caching=true; /开启 缓存 
$smarty->cache dir=BASE PATH.SMARTY PATH.cache/: /定义 组 存 文件 存储 位 置 

2. 设置 缓存 的 生命 周期 
| 缓存 创建 成 功 后 ， 还 必须 为 它 设置 一 个 生命 周期 如 果 它 一 直 不 更 新 ， 那 么 就 没有 任何 意 
| 义 。 设 置 缓存 生命 周期 应 用 的 是 Smarty 对 象 中 的 $cache_lifetime 属性 ， 缓 存 时 间 以 秒 为 单位 ， 
| 默认 值 是 3600 秒 。 其 操作 代码 如 下 : 


$smarty->caching=true; /开启 缓存 
$smarty->cache_ dir= BASE PATH.SMARTY PATH.cache/: /定义 缓存 文件 存储 位 置 
$smarty->cache lifetime=3600 // 设 置 缓存 时 间 为 1 小 时 


3， 判断 模板 文件 是 否 已 经 被 缓存 
| 如 果 页 面 已 经 被 缓存 ， 那 么 就 可 以 直接 调用 缓存 文件 ， 而 不 再 执行 动态 获取 数据 和 输出 的 
| 操作 。 为 了 避免 在 开启 缓存 后 再 次 执行 动态 获取 数据 和 输出 操作 给 服务 器 带 来 的 压力 ， 最 佳 的 
| 方法 就 是 应 用 Smarty 对 象 中 的 is_cached() 方 法 判断 指定 的 模板 是 否 存在 缓存 , 如 果 存 在 则 直接 
| 执行 缓存 中 的 文件 ， 和 否则 执行 动态 获取 数据 和 输出 的 操作 。 操 作 代 码 如 下 : 
| $smarty->caching=true; /开启 缓存 
if(!$smarty->is_cached('index html’)){ 
// 执 行动 态 获取 数据 和 输出 的 操作 
; 
$smarty->display(index.html); 
4 清除 模板 中 的 缓存 
| 。。 缓存 的 清除 有 两 种 方法 : 
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第 一 种 是 clear all_cache() 方 法 ， 清 除 所 有 模板 缓存 。 其 语法 如 下 : 
void clear all cache (int expire time) | 
可 选 参数 expire time， 可 以 指定 一 个 以 秒 为 单位 的 最 小 时 间 ， 超过 这 个 时 间 的 缓存 都 将 被 
清除 。 
第 二 种 是 clear_cache() 方 法 ， 清 除 指定 模板 的 缓存 。 
Void clear_ cache (string template [, string cache id [, string compile id [, int expire time]]]) 
如 果 这 个 模板 有 多 个 缓存 ， 则 既 可 以 用 第 二 个 参数 指定 要 清除 缓存 的 缓存 号 ， 也 可 以 通过 | 
第 三 个 参数 指定 编译 号 。 可 以 把 模板 分 组 ， 以 便 可 以 方便 地 清除 一 组 缓存 。 第 四 个 参数 是 可 选 | 
的 ， 用 来 指定 超过 某 一 时 间 (以 秒 为 单位 〉 的 缓存 才 会 被 清除 。 | 


5. 应 用 示例 

在 本 示例 中 开启 网 站 注册 页 面 的 缓存 。 具 体操 作 在 index.php 文件 中 完成 ， 其 代码 如 下 : 
<?php 
header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
require_once("system/system.inc.php"); // 包 含 配置 文件 
S$smarty->caching=true; /开启 缓存 
$smarty->cache lifetime=3600; // 设 置 缓存 过 期 时 间 
//$smarty->clear all_cache(); // 清 除 所 有 缓存 
if(!$smarty->is_cached("index.html")){ 
S$res=$conn->execute("select * from tb_bccd "): /执行 select 查询 语句 
Sarray=$res->GetArray(); 
$array_id=array(); 


S$array_name=array(): 

for($i=0;$i<$res->RecordCount():$i++){ 
$array_id[]=$array[$i][id]: 
$array_name[]=$array[$i][name]; 


} 


$smarty->assign('cust_id', $array_id): // 设 置 下 拉 列 表 的 值 
$smarty->assign('cust_name', $array_name):; // 设 置 列表 的 显示 数据 

} 

$array= explode( ', mt_rand(1000,9999)): // 生 成 随机 验证 码 
$smarty->assign(title', 开 启 网 站 注册 页 面 的 缓存 ); // 将 指定 数据 赋 给 模板 变量 
$smarty->assign('content', $array): // 将 数组 赋 给 模板 变量 
$smarty->assign('contents'.$arrays): // 将 数组 赋 给 模板 变量 
$smarty->display('index.html’): // 指 定 模 板 页 

?> 


缓存 文件 存储 在 system\Smarty\cache 目录 下 , 而 存储 路 径 的 设置 是 在 systemvsystem smarty | 
-inc.php 文件 中 完成 的 。 | 


问题 264 如何 判断 同一 模板 中 的 多 个 绥 存 文件 ? 


问题 阐述 


在 实际 的 程序 开发 中 经 常会 遇 到 这 样 的 情况 : 同一 个 模板 文件 生成 多 个 页 面 。 那么 就 要 生成 
多 个 缓存 文件 ,这 多 个 缓存 文件 是 如 何 生成 的 呢 ? 又 如 何 判断 同一 模板 中 的 多 个 缓存 文件 呢 ? 
.415 。 Le 
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| 多 个 页 面 进行 缓存 应 用 的 是 Smarty 中 的 display0 方 法 , 通过 该 方法 的 第 一 个 参数 设置 缓存 
久 | 号 ， 有 几 个 不 同 的 缓存 号 就 有 几 个 缓存 页 面 。 操 作 代码 如 下 ; 

| $smarty->display(Cindex.html,$ GET[idT): /将 id 作为 第 二 个 参数 传递 
判断 同一 模板 中 的 多 个 缓存 是 否 存在 ， 其 依据 的 是 缓存 号 。 判 断 同 一 模板 的 多 个 缓存 是 理 
| 存在 可 应 用 is_cached0 方 法 , 通过 该 方法 的 第 二 个 参数 设置 缓存 号 , 判断 对 应 的 缓存 是 否 存在 。 


| 其 方法 如 下 : 

| $smarty->caching=true; /开启 缓存 
$smarty->cache dir=BASE PATH.SMARTY PATH.'cache/; /定义 缓存 文件 存储 位 置 
$smarty->cache lifetime=3600: /设置 缓存 时 间 为 1 小 时 
if(!$smarty->is cached(index html',$ GET['d'])){ 

// 执 行动 态 获取 数据 和 输出 的 操作 

bh 
$smarty->display(index.html,$_GET[id]); /将 id 作为 第 二 个 参数 传递 


问题 265 ”如何 通过 配置 文件 定义 变量 ? 


”问题 阐述 
| 配置 文件 的 应 用 有 利于 设计 者 管理 文件 中 的 模板 全 局 变量 。 例 如 , 定义 一 个 模板 色彩 变量 。 
| 一 般 情况 下 如 果 想 改变 一 个 程序 的 外 观 色 彩 ， 则 必须 更 改 每 一 个 文件 的 颜色 变量 。 如 果 有 配置 
| 文件， 色彩 变量 就 可 以 保存 在 一 个 单独 的 文件 中 ， 只 要 改变 配置 文件 就 可 以 实现 色彩 的 更 新 ， 
| 这 与 在 Web 页 面 开发 中 应 用 的 CSS 样式 非常 相似 ， 那 么 如 何 通过 配置 文件 定义 变量 呢 ? 
| 专家 解答 
| “通过 配置 文件 定义 变量 主要 有 以 下 3 个 步 又。 

1. 创建 配置 文件 
| 配置 文件 可 以 任意 命名 , 其 存储 位 置 由 Smarty 对 象 的 Sconfig dir 属性 指定 。 如 果 存 在 不 只 
| 在 一 个 区 域内 使 用 的 变量 值 , 可 以 使 用 三 引号 (""") 将 它 完整 地 封装 起 来 。 在 创建 配置 文件 时 ， 
| 建议 在 程序 运行 前 使 用 “#” 加 一 些 注释 信息 ， 这 样 有 助 于 程序 的 阅读 、 更 新 。 
| 在 配置 文件 中 既 可 以 声明 全 局 变量 ， 也 可 以 声明 局 部 变量 。 如 果 声 明 局 部 变量 ， 则 可 以 使 
| 用 中 括号 〈[]) 括 起 来 ， 在 中 括号 之 内 声明 的 变量 属于 局 部 变量 ， 而 中 括号 之 外 声明 的 变量 都 
| 是 全 局 变量 。 中 括号 的 使 用 不 仅 使 配置 文件 中 声明 变量 的 模块 变 得 清晰 ， 而 且 可 以 在 模板 中 选 
| 择 加 载 中 括号 内 的 变量 。 
| 2， 加 载 配 置 文件 

加 载 配置 文件 应 用 Smarty 的 内 建 函数 config load， 其 语法 如 下 : 

{config load file="file name " section="add attribute" scope="" global=""} 

参数 说 明 : 

| @ file: 指定 包含 的 配置 文件 的 名 称 。 
了 .416 。 
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@ section: 附加 属性 ， 当 配置 文件 中 包含 多 个 部 分 时 应 用 ， 指 定 具 体 从 哪 一 部 分 中 取得 变量 。 | 
@ scope: 加 载 数据 的 作用 域 ， 取 值 必须 为 local、parent 或 global。local 说 明 该 变量 的 作用 | 
域 为 当前 模板 ，parent 说 明 该 变量 的 作用 域 为 当前 模板 和 当前 模板 的 父 模板 (调用 当前 模板 的 | 
模板 )，global 说 明 该 变量 的 作用 域 为 所 有 模板 。 | 
@ global: 说 明 加 载 的 变量 是 否 全 局 可 见 ， 等 同 于 scope=parent。 


提示 : 
当 指定 scope 属性 时 ， 可 以 设置 global 属性 ， 但 模板 忽略 该 属性 值 ， 而 以 scope 属性 为 准 。 | 
3. 引用 配置 文件 中 的 变量 | 
配置 文件 加 载 成 功 后 ， 就 可 以 在 模板 中 引用 配置 文件 中 声明 的 变量 。 引 用 配置 文件 应 用 的 | 
是 “#” 或 者 Smarty 的 保留 变量 $smarty.config。 其 应 用 示例 如 下 : | 
{ config load file="file con.conf"} 从 加 载 配置 文件 *} 
{#title## 
<td height="228" colspan="2" align="left" valign="top" class="{S$smarty.config.styles}"> 
4 应 用 示例 | 
(1) 创建 system 文件 夹 ， 定 义 Smarty 文件 夹 ， 存 储 编译 目录 、 缓 存 目 录 和 配置 文件 目 | 
录 ; 创建 类 文件 system.smarty.inc.php， 定 义 Smarty 的 配置 类 ， 在 该 类 中 指定 配置 文件 存储 的 
目录 ， 定 义 类 的 实例 化 文件 system.inc php， 完 成 Smarty 类 的 实例 化 操作 ， 并 返回 操作 对 象 。 | 
(2) 创建 index.php 文件 ， 载 入 类 的 实例 化 文件 ;通过 文件 系统 函数 file_get_contents(0) 读 | 
取 文 本 文件 中 的 数据 ， 并 且 将 数据 存储 到 模板 变量 中 ， 最 终 指定 模板 页 。 其 代码 如 下 : | 


<2php 

header ( "Content-type: text/html: charset=UTF-8" ); /设置 文件 编码 格式 
include("system/system.inc.php"): // 包 含 配置 文件 

$str = file_get_contents('files/content.txt'):; // 读 取 文 本 文件 中 的 数据 | 
$smarty->assign('content ,iconv("gb2312","utf-8",$str)); /将 文本 文件 中 的 数据 存储 到 模板 变量 中 | 
$smarty->display('index.html'); // 指 定 模板 页 | 
?> 


(3) 创建 index html 模板 页 。 首 先 ， 通 过 config load0 函 数 加 载 配置 文件 。 然后 , 应 用 “#”| 
和 Ssmarty.config 引用 配置 文件 中 的 变量 。 最 后 ， 通 过 模板 变量 输出 文本 文件 中 的 数据 。 其 关键 | 
代码 如 下 : | 


{ config load file="file_con.conf"} 化 加 载 配置 文件 *} 

<title>{#title 内 </title> | 

<body leftmargin="{#leftmargin#}" topmargin="{#topmargin#}" marginwidth="{#marginwidth#}" | 
marginheight=" {#marginheight#}"> | 

<td height="228" colspan="2" align="left" valign="top" class=" {$smarty.config.styles}">{$content} </td> 


问题 266 ”在 Smarty 模板 中 ， 如 何 截 取 中 文字 符 串 ? 


问题 讲述 | 
在 PHP 中， 可 以 通过 自 定义 函数 来 完成 对 中 文字 符 串 的 截取 ， 那 么 在 Smarty 模板 中 ， 如 
。417 。 四 
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as 
tt 向 
何 截取 中 文字 符 串 呢 ? 
专家 解答 
| 在 Smarty 模板 中 提供 了 truncate 方法 对 字符 串 进行 截取 ， 该 方法 会 截取 到 一 个 词 的 末尾 ， 
| 直接 避免 了 在 截取 中 文字 符 串 时 出 现 乱码 的 问题 。truncate 方法 的 参数 说 明 如 表 14.4 所 示 。 
表 14.4 truncate 方法 的 参数 说 明 

说 明 
第 一 个 参数 ， 指 定 截取 字符 的 数量 ， 类 型 为 Integer 
第 二 个 参数 ， 指 定 截取 后 追加 在 截取 词 后 面 的 字符 串 ， 类 型 为 String 
第 三 个 参数 ， 类 型 为 Boolean， 如 果 该 值 为 false， 表 示 截 取 到 词 的 边界 ， 如 果 该 值 为 tue， 
表示 截取 时 精确 到 指定 字符 


应 用 示例 
| 在 本 示例 中 ， 应 用 truncate 方法 对 论坛 中 的 标题 和 内 容 进行 截取 ， 并 且 应 用 省 略 号 替换 截 
取 的 内 容 。 其 具体 步骤 如 下 : 

(1) 创建 system 文件 夹 ， 定 义 Smarty 文件 夹 ， 存 储 编译 目录 、 缓 存 目录 ; 创建 类 文件 
system.smarty.inc.php， 定 义 数据 库 连 接 、 操 作 、 分 页 和 Smarty 的 配置 类 ， 定 义 类 的 实例 化 文件 
system.inc.php， 完 成 各 个 类 的 实例 化 操作 ， 并 返回 操作 对 象 。 

(2) 创建 index.php 文件 ， 载 入 类 的 实例 化 文件 ， 分 页 读 取 数据 库 中 存储 的 论坛 数据 ， 并 
且 将 读 取 的 结果 存储 到 模板 变量 中 ， 最 终 指定 模板 页 。 其 代码 如 下 : 

<2php 
require_once("system/system.inc.php"); // 调 用 指定 的 文件 
$array=$seppage->ShowDate("select*from tb_guestbook order by id desc",$conn,6,$_GET["page"]); 
// 调 用 分 页 类 ， 实 现 分 页 功能 
if(!$array){ 
$smarty->assign("iscommo","F"); /判断 如 果 执 行 失败 ， 则 输出 模板 变量 iscommo 的 值 为 F 
}else{ 
$smarty->assign("iscommo","T"); /判断 如 果 执 行 成 功 ， 则 输出 模板 变量 iscommo 的 值 为 了 
$smarty->assign("showpage",$seppage->ShowPage(" 帖子 "" 条 ",","al ")): 
// 定 义 输出 分 页 数据 的 模板 变量 showpage 


$smarty->assign("array",$array): 
| } 
| $smarty->assign('title','truncate 方法 截取 字符 串 "); /定义 标题 
$smarty->display('index.html'); // 指 定 模板 页 
?> 


(3) 创建 index.html 模板 页 ， 获 取 模 板 变 量 传递 的 数据 ， 实 现 数据 的 分 页 输出 ， 并 且 通 过 
| truncate 方法 对 输出 的 标题 和 内 容 进行 截取 操作 ， 标 题 截取 30 个 字 节 ， 内 容 截取 60 个 字 节 ， 
| 应 用 省 略 号 补 齐 截取 部 分 。 其 关键 代码 如 下 : 
| {section name=id loop=$array} 
| <tr> 
<td> 
| {$array[id].id} {$array[id].titleltruancate:30:"...":false}:; 
| 积分 {$array[id] integral} 
| 时 间 : {$array[id].createtimeltruncate:12:""}<br> 
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{$array[id].contentltruncate:60:”..……":false} 
</td> 
</tr> 
{/section} 


运行 结果 如 图 14.14 所 示 。 
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图 14.14 应 用 tmuncate 方法 截取 字符 串 
专家 点 评 | 
在 Smarty 模板 中 应 用 truncate 方法 截取 中 文字 符 串 时 同样 会 出 现 乱码 的 问题 。 要 想 解 决 这 
个 问题 ,就 要 定位 到 Smarty 类 库 pluginsmodifiertruncate php 文件， 重新 编辑 smarty modifier | 
truncate 方法 ， 将 其 中 截取 字符 串 使 用 的 substr0 函 数 全 部 替换 为 mb_substr0 函 数 。 这 样 在 调用 | 
truncate 方法 截取 中 文字 符 串 时 就 不 会 出 现 乱 码 的 问题 了 。 | 


问题 267 在 Smarty 中 如 何 进 行 编码 ? 


问题 阐述 
在 Smarty 模板 中 如 何 进 行 编码 ? 
专家 解答 


在 PHP 脚本 中 应 用 urlencode() 函 数 对 超 链 接 中 传递 的 参数 进行 编码 ， 而 在 Smarty 模板 中 | 
提供 escape 函数 对 字符 串 进 行 编码 。 | 
escape 变量 用 于 HTML 转 码 、url 转 码 ， 在 没有 转 码 的 变量 上 转换 单 引 号 、 十 六 进 制 转 码 | 
或 JavaScript 转 码 。 默 认 是 HTML 转 码 。 其 基本 的 应 用 格式 如 下 : | 
{$articleTitlelescape} 
{$articleTitlelescape:"html"} {*escapes &"'<>*} 
{$articleTitlelescape:"htmlall"} {* escapes ALL html entities *} 
{$articleTitlelescape:"url"} 
{$articleTitlelescape:"quotes"} 
<a href="mailto: {$EmailAddresslescape:"hex"}">{$EmailAddresslescape:"hexentity"}</a> 


应 用 示例 | 
在 本 示例 中 应 用 escape 函数 对 Smarty 模板 页 中 超 链接 传递 的 参数 进行 编码 。 其 关键 步骤 | 
如 下 : | 
(1) 在 main.html 模板 页 中 为 不 同 模块 之 间 跳 转 的 超 链接 传递 的 参数 值 进行 编码 ， 其 关键 | 
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! EC 
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| II 
| 代码 如 下 : 

| <map name="Map" id="Map"> 

| <area shape="rect" coords="29.41,88.62 "href="main.php?caption={" 商 品 添加 "|escape:"url"} &amp; 
| type= 人 "商品 管理 "}" 户 


食 关 | | <area shape="rect" coords="30,71,91,90 "href="main.php?caption={" 商 品 修改 "|escape:"url"} &amp: 
| type= 人 "商品 管理 "}" /> 
| type={" 商 品 管理 "}" /> 
| </map> 


<area shape="rect" coords="31.99.91,118 "href="main.php?caption={" 商 品 市 除 "|escape:"url"} &amp; 

| <map name="Map2" id="Map2"> 
| <area shape="rect" coords="30,45,97,63"” href="main.php?caption={" 会 员 删 除 "|escape:"url"} &amp: 
| gpesf "会员 管 理 "" 亡 
| <area shape="rect" coords="34,80,89,98" href="main.php?caption={" 会 员 浏 览 "lescape:"url"} &amp; 
| type={" 会 员 管 理 ")}" 放 > 
| </map> 

<map name="Map5" id="Map5"> 

<area shape="rect" coords="12,11,54,36" href="logout.php" /> 

</map> 

<map name="Map6" id="Map6"> 
| <area shape="Tect" coords="14,12,51,36" href="main.php?caption={" 商 品 修改 "|escape:"url"}&amp; 
| type={" 商 品 管理 "lescape:"url"}" /> 
| </map> 


| (2) 虽然 通过 escape 编码 后 的 模板 变量 输出 时 是 乱码 ， 但 是 在 switch 语句 中 仍然 可 以 直 
， 接 使 用 $_GET['caption"] 获 取 超 链接 的 值 , 根据 这 个 值 作出 判断 , 在 主页 中 输出 哪个 模块 的 内 容 。 
| 此 时 不 需要 对 该 参数 值 进行 解码 ， 且 需 注意 的 是 , 执行 判断 是 在 main.php 文件 中 完成 的 。 其 关 


| 键 代码 如 下 : 
| <?php 

session _startO: /初始 化 SESSION 变量 

这 $_SESSION[mser]!="" and $_ SESSION[pass']!=""){ ”// 判 断 用 户 名 和 密码 是 否 为 空 

require("system/system.inc.php"): // 包 含 PHP 配置 文件 

switch ($_GET['caption']){ // 根 据 超 链 接 传递 的 值 进行 判断 

case "商品 添加 "; 
include "sho_insert php": /包含 PHP 文件 
$smarty->assign(admin phtml',sho_inserthtml):// 将 PHP 文件 对 应 的 模板 文件 的 名 称 赋 

| 给 模板 变量 
| break; // 跳 出 循环 
| /省 略 了 部 分 代码 
| default: 
| include "sho_update.php"; 
| $smarty->assign(admin phtml','sho_update.html’); 
| break: 
) 
| $smarty->assign("title"," 后 台 管 理 系 统 --".$_GET['caption"]);// 为 模板 变量 赋值 
| S$smarty->assign("caption",$_GET['caption']); // 为 模板 变量 赋值 ， 输 出 模块 名 称 
| $smarty->assign("type",$_GETT['type']); // 为 模板 变量 赋值 ， 输 出 模块 所 属 类 别 

$smarty->assign("user",$_SESSION[user]): /为 模板 变量 赋值 ， 输 出 登录 用 户 名 称 
| $smarty->display("main.html"): // 指 定 模板 页 
| jelse{ 
| echo "<script>alert( 您 不 具备 访问 权限 ! '); windowlocation hre 全 index.html:</scripf> 
】 “420 
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运行 结果 如 图 14.15 所 示 。 
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图 14.15 ”Smarty 模板 中 对 超 链 接 的 参数 值 进 行 编码 


经 过 escape 变量 编码 后 的 字符 串 在 输出 时 将 是 一 堆 乱 码 , 在 超 链接 中 通过 这 种 方法 可 以 保 
证 参数 值 的 安全 传递 。 


问题 268 ”如 何在 Smarty 中 定义 注册 模板 函数 ? 


问题 曾 述 
如 何在 Smarty 中 定义 注册 模板 函数 ? 

专家 解答 | 
模板 函数 类 似 于 PHP 中 的 自 定义 函数 ， 特 点 是 其 注册 在 动态 页 中 完成 ， 而 实际 的 应 用 在 | 


静态 页 中 进行 ， 完 全 体现 了 Smarty 的 动静 分 离 的 开发 模式 ， 是 在 Smarty 模板 中 实现 一 些 特殊 
功能 的 首选 方法 。 
1. register_function 方法 
register_function 是 Smarty 模板 中 的 动态 注册 模板 函数 插件 。 其 语法 如 下 : 
void register_function (string name, mixed impl, bool cacheable, array or null cache_attrs) 


register_function 方法 的 参数 说 明 如 表 14.5 所 示 。 


表 14.5 register_function 方法 的 参数 说 明 
参数 说 了 明 
name 模板 函数 的 名 称 
执行 函数 的 名 称 。 执 行 函数 的 格式 可 以 是 一 个 包含 函数 名 称 的 字符 串 ; 也 可 以 是 一 个 
array(&$object $method) 数 组 形式 ， 其 中 &$object 是 一 个 对 象 的 引用 ， 而 $method 是 它 的 一 个 
方法 ; 还 可 以 是 一 个 array(&$ class, $method) 数 组 形式 , 其 中 $class 是 一 个 类 的 名 称 , $method 
是 类 中 的 一 个 方法 
cacheable | 可 选 参 数 ， 在 大 多 数 情 况 下 可 以 省 略 
cache attrs | 可 选 参数 ， 在 大 多 数 情况 下 可 以 省 略 
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2.，ExtractO 函 数 
Extract(0) 函 数 从 数组 中 将 变量 导入 到 当前 的 符号 表 中 。 其 语法 如 下 : 
int extract ( array var_array [, int extract type [, string prefix]] ) 
本 函数 将 变量 从 数组 中 导入 到 当前 的 符号 表 中 。 接 受 结合 数组 var_array 作为 参数 并 将 键 
名 当 作 变量 名 ， 值 作为 变量 的 值 。 对 每 个 键 / 值 对 都 会 在 当前 的 符号 表 中 建立 变量 ， 并 受到 
extract type 和 prefix 参数 的 影响 。 其 返回 值 是 成 功 导入 到 符号 表 中 的 变量 数目 。 


(1) 自 版 本 PHP 4.0.5 起 本 函数 返回 被 提取 的 变量 数目 . (2) EXTR IF EXISTS 和 EXTR_ 


| PREFIX IF EXISTS 是 版 本 PHP 4.2.0 中 引进 的 . (3 ) EXTR REFS 是 版 本 PHP 4.3.0 中 引进 的 。 


ExtractO 函 数 检查 每 个 键 名 ， 看 是 否 可 以 作为 一 个 合法 的 变量 名 ， 同 时 也 检查 和 符号 表 中 
已 有 的 变量 名 的 冲突 。 对 待 非 法 数字 和 冲突 的 键 名 的 方法 将 根据 extract type 参数 决定 。 可 以 
是 以 下 值 之 一 : 

EXTR_OVERWRITE: 如 果 有 冲突 ， 履 盖 已 有 的 变量 。 

EXTR_SKIP: 如 果 有 冲突 ， 不 覆盖 已 有 的 变量 。 

EXTR _ PREFIX SAME: 如 果 有 冲突 ， 在 变量 名 前 加 上 前 缀 prefix 。 

EXTR_PREFIX_ALL: 给 所 有 变量 名 加 上 前 级 prefix。 自 PHP4.0.5 起 这 也 包括 了 对 
数字 索引 的 处 理 。 

EXTR_PREFIX_INVALID: 仅 在 非法 数字 的 变量 名 前 加 上 前 缀 prefix。 本 标记 是 PHP 
4.0.5 新 加 的 。 

回 EXTR_ IF EXISTS: 仅 在 当前 符号 表 中 已 有 同名 变量 时 覆盖 它们 的 值 ， 其 他 的 都 不 
处 理 。 可 以 用 在 已 经 定义 了 一 组 合法 的 变量 , 然后 要 从 一 个 数组 ， 如 $_REQUEST 中 
提取 值 绪 盖 这 些 变量 的 场合 。 本 标记 是 PHP 4.2.0 新 加 的 。 

回 EXTR PREFIX IF EXISTS: 仅 在 当前 符号 表 中 已 有 同名 变量 时 建立 附加 了 前 级 的 
变量 名 ， 其 他 的 都 不 处 理 。 本 标记 是 PHP 4.2.0 新 加 的 。 

回 ”EXTR REFS: 将 变量 作为 引用 提取 。 这 有 力 地 表明 了 导入 的 变量 仍然 引用 了 
var_array 参数 的 值 。 可 以 单独 使 用 这 个 标志 或 者 在 extract type 中 用 OR 与 其 他 任何 
标志 结合 使 用 。 本 标记 是 PHP 4.3.0 新 加 的 。 

回 ”如 果 没 有 指定 extract type， 则 被 假定 为 EXTR_OVERWRITE。 


注意 : 

prefix 仅 在 extract type 的 值 是 EXTR PREFIX SAME、EXTR PREFIX ALL、EXTR_ 
PREFIX_INVALID 或 EXTR_PREFIX IF EXISTS 时 需要 。 如 果 附加 了 前 级 后 的 结果 不 是 合法 
的 变量 名 ， 将 不 会 导入 到 符号 表 中 。 


3. 应 用 示例 

在 本 示例 中 通过 注册 模板 函数 对 帖子 的 内 容 进行 截取 ,如 果 帖子 内 容 超出 指定 的 长 度 就 对 
其 进行 截取 ， 并 且 使 用 省 略 号 替代 被 截取 部 分 。 其 具体 步骤 如 下 : 

(1) 创建 system 文件 夹 ， 定 义 Smarty 文件 夹 ， 存 储 编 译 目录 、 缓 存 目 录 ， 创建 类 文件 
system.smartyinc.php， 定 义 数 据 库 连 接 、 操 作 、 分 页 和 Smarty 的 配置 类 ， 定 义 类 的 实例 化 文件 
system.inc.php， 完 成 各 个 类 的 实例 化 操作 ， 并 返回 操作 对 和 象 。 
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(2) 创建 index.php 文件 ， 载 入 类 的 实例 化 文件 ; 编写 自 定义 函数 unHtml， 对 论坛 中 的 内 
容 进行 截取 操作 ;应 用 register_function 注册 模板 函数 ; 分 页 读 取 数 据 库 中 存储 的 论坛 数据 ， 
并 且 将 读 取 的 结 

果 存 储 到 模板 变量 中 ， 最 终 指定 模板 页 。 其 代码 如 下 : 


<?php 
header ( "Content-type: text/html:; charset=UTF-8" ); // 设 置 文件 编码 格式 


require_once("system/system.inc.php"); // 调 用 指定 的 文件 
fnnction unHtml($params) { /创建 模板 函数 | 
extract($params); // 读 取 数 据 | 
if(strlen($text)>40){ | 
$str=mb_substr($text,0,40,"utf-8")."..."; 1/ 截取 字符 串 | 
Jelse{ | 
$str=$text; | 
} | 
return $str; // 返 回 截取 结果 | 
} | 
$smarty->register function("Util", "unHtml"): // 注 册 模 板 函 数 | 


$array=$seppage->ShowDate("select * from tb_ guestbook order by id desc",$conn,6,$_GET["page'"]); | 
/调用 分 页 类 ， 实 现 分 页 功能 | 
if(!$array){ 
$smarty->assign("iscommo","F");// 判 断 如 果 执 行 失败 ， 则 输出 模板 变量 iscommo 的 值 为 F 
}else{ 


$smarty->assign("iscommo","T");// 判 断 如 果 执行 成 功 , 则 输出 模板 变量 iscommo 的 值 为 T | 
$smarty->assign("showpage",$seppage->ShowPage(" 帖 子 "," 条 ","","al1")); | 
/定义 输出 分 页 数据 的 模板 变量 showpage 
$smarty->assign("array", $array): 
} 
$smarty->assign('"title','register_function0 方 法 注册 模板 函数 '); | 
$smarty->display('index.html"); | 
Ts 
(3) 创建 index.html 模板 页 ， 获 取 模板 变量 传递 的 数据 ， 实 现 数据 的 分 页 输出 ， 并且 调 用 | 
动态 页 中 注册 的 模板 函数 实现 对 论坛 中 内 容 的 截取 操作 。 其 关键 代码 如 下 : | 
{section name=id loop=$array} 
<tr> 
<td colspan="4" bgcolor="#FFFFFF">{$array[id].id }&nbsp:&nbsp:{fUtil text=$array[id].title } 积 分 : 
{$array[id].integral} &nbsp;&nbsp:&nbsp;&nbsp: 时 间 : {$araylid].createtime} <br>{Util text=$array[lid].content} </td> 
</tr> 
{/section} 
运行 结果 如 图 14.16 所 示 。 | 
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图 14.16 注册 模板 函数 截取 字符 串 | 
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问题 269 ”如 何在 Smarty 中 设计 页 面 ? 


俩 丰 。 问 题 阅 述 
在 Smarty 模板 页 中 ， 如 何 对 页 面 进行 设计 ? 


| 专家 解答 
| 在 Smarty 模板 页 中 同样 可 以 将 网 页 的 头 、 尾 和 主 文件 进行 分 别 存储 ， 从 而 达到 页 面 重用 
| 的 效果 。 


| 在 Smarty 模板 页 中 应 用 include 函数 载 入 模板 页 。include 函数 用 于 在 当前 模板 中 包含 其 他 
| 模板 ， 当 前 模板 中 的 变量 在 被 包含 的 模板 中 可 用 。include 函数 语法 如 下 : 
| {include file="file name " assign=" " var=" "} 


| 参数 file 指定 包含 模板 文件 的 名 称 ， 为 必 先 参数， 参数 assign 指定 一 个 变量 保存 包含 模板 
| 的 输出 ;参数 var 传递 给 待 包 含 模板 的 本 地 参数 ， 只 在 待 包 含 模板 中 有 效 。 

| 应 用 示例 

| 本 示例 中 应 用 include 函数 在 模板 文件 中 完成 网 页 结构 的 设计 ， 实 现 网 页 头 、 尾 和 主 文件 
| 的 分 离 。 其 实现 步骤 如 下 : 

| (1) 创建 system 文件 夹 ， 封 装 Smarty 模板 的 配置 方法 ， 这 里 不 再 效 述 。 

| (2) 创建 index.php 文件 ， 应 用 switch 语句 根据 超 链接 传递 的 变量 值 ， 包 含 不 同 的 PHP 
”脚本 文件 ， 同 时 将 对 应 的 模板 页 名 称 赋 给 指定 的 模板 变量 。 其 代码 如 下 : 


<2php 
require("system/system.inc.php"): // 载 入 配置 文件 
switch ($_GET['caption']){ /应 用 switch 语句 ， 进 行 判断 
case "Treg" 
include "reg php"; // 包 含 PHP 脚本 
$smarty->assign('admin phtml','reg.html'): // 通 过 模板 变量 指定 模板 页 
break: 
case "log"; 
include "log.php"; 
$smarty->assign(admin phtml','log.html"); 
break: 
case "from"; 
include "for.php"; 
$smarty->assign(admin_ phtml','for.html"); 
break:; 


include "mes.php"; 
$smarty->assign(‘admin phtml','mes.html’); 
break; 
default: 
include "forphp": 
$smarty->assign(admin _ phtml,forhtml): 
break; 


司 .424 . 


第 ] 4 章 Smarty 模 极 Ye 
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$smarty->assign("title","Smarty 模板 中 的 页 面 设计 --".$_GET[caption]); 
$smarty->display("index.html"); /指定 模板 页 
?> 


(3) 创建 indexhtml 模板 页 ， 应 用 include 函数 载 入 页 面 的 头 文件 (tophtml)、 尾 文件 | 
《bottom.html》 和 主 文件 。 其 关键 代码 如 下 : | 

fgite) 
{include file=top.html} 
{include file=$admin phtml} 
{include file=right.html} 
{include file=bottom html} 

(4) 创建 include 函数 加 载 的 模板 文件 和 动态 PHP 文件 。 

运行 结果 如 图 14.17 所 示 。 


FANXING -xs so /sn ea 
SD) CT 到 名 用 片 文件 夫 
毛 3 
口 。。 术 要 位 置 基 外 uo 1s0 
本 各 放下 方式 
口 。 标题 位 和 雪村 处 ui0 20 多 LN 
口 。 标本 位置 交 于 处 ute a3s0 = 
四。 本 位置 雪 处 uo 0 Ws ee 
加 标 晤 位 置 寺 处 ii0 aa < 
口 奈 取 位 垩 芍 普 处 4140 3330 
口 标 墅 位置 芍 置 处 no 20 
口 。 要 要 位 团委 处 uo 
共有 2 页 ,当前 是 问 责 [| 


图 14.17 Smarty 模板 中 的 页 面 设计 
专家 点 评 | 
在 程序 设计 中 ， 所 有 依据 条 件 做 出 判定 的 问题 都 可 以 用 站 条 件 语 句 来 解决 。 不 过 ， 在 用 | 
f...else 语句 处 理 多 个 条 件 的 判定 问题 时 ， 组 成 条 件 的 表达 式 在 每 一 个 elseif 中 都 要 计算 一 次 ， | 
显得 繁琐 爱 肿 。 为 了 避免 站 语句 的 见长 ， 提 高 程序 的 可 读 性 ， 可 以 使 用 switch 分 支 控制 语句 。 | 
switch 语句 的 语法 如 下 : | 


switch(variable){ 
case valuel: 


default: 
default statement n: 
} 


switch 语句 根据 variable 的 值 依次 与 case 中 的 value 值 相 比 较 ， 如 果 不 相 等 ， 继 续 查 找 下 
一 个 case; 如 果 相 等 , 就 执行 对 应 的 语句 ， 直 到 switch 语句 结束 或 遇 到 break 为 止 。 一 般 switch | 
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Ca 
让 和 
语句 最 终 都 有 一 个 默认 值 default, 如 果 在 前 面 的 case 中 没有 找到 相符 的 条 件 , 则 输出 默认 语句 ， 
| 和 else 语句 类 似 。 


问题 270 ”如 何在 Smarty 中 制作 日历 ? 


| 问题 阐述 
如 今 很 多 网 站 都 有 在 线 日 历 功能 , 如 Google、Yahoo 等 。 那么 在 Smarty 中 如 何 制 作 日 历 呢 ? 
| 专家 解答 


在 Smarty 模板 中 制作 日 历 将 应 用 到 如 下 技术 : 
(1) 通过 {html table} 函 数 生成 日 历 表格 ，{html_table} 函 数 根 据 指定 的 列 数 ， 生 成 规定 列 
的 表格 。 其 语法 如 下 : 


| 表 14.6 Te 


属 性 | 类 型 | 是 否 必须 | 默认 值 | 描述 

loop | 待 沉 历 的 数组 

cols | integer | No | 3 | 表格 的 列 数 日 

table attr | | 表格 的 属性 

tr attr EE a 行 标签 属性 (或 轮转 数组 ) 

tdattr | sting | No | empty | 列 标签 属性 (或 轮转 数组 ) 

trailpad | sting | No | &nbsp， | 最 后 一 行 附加 的 数据 (如 果 有 的 话 ) 
hdir | sting | No | right | 行 的 对 齐 方式 ， 可 能 的 值 为 left 或 right 
vdir | sting | No | down | 列 的 对 齐 方式 ， 可 能 的 值 为 up 或 down 


| 自 定义 函数 {html_table} 将 数组 中 的 数据 填充 到 HTML 表格 中 。cols 属性 决定 表格 有 多 少 列 ; 
| table attr、tr_attr 和 td_attr 属性 决定 表格 中 tt 和 也 标签 的 一 些 附 加 属性 ， 如 果 tr_attr 和 td_attr 属 
， 性 值 为 数组 ,将 轮流 使 用 其 中 的 值 ， 如 果 指定 了 trailpad 属性 , 将 在 表 尾 最 后 一 行 附 加 一 些 数 据 。 
| (2) 在 动态 PHP 文件 中 通过 自 定义 函数 生成 日 历数 据 ， 并 且 将 生成 的 数据 通过 模板 变量 
| 传递 到 模板 页 中 ， 在 {html table} 函 数 生成 的 表格 中 输出 日 历数 据 。 自 定义 函数 的 语法 如 下 : 


| function get total day($date){ // 自 定义 函数 
| $time_stamp=strtotime($date): // 获 取 当 前 时 间 的 时 间 截 
| $month ar=explode("/",$date): // 对 当前 时 间 进 行 拆 分 
| $month=$month_ar[0]; // 将 拆 分 结果 直接 的 月 份 值 赋 给 指定 变量 
| $year=$month ar[2]: /将 拆 分 结果 值 赋 给 指定 的 年 份 变量 
| echo $year ."-". $month: // 输 出 当前 的 年 、 月 值 
| $days=date("j",mktime(0,0,0,$month+1,0,$year)); /获取 当前 时 间 中 本 月 的 天 数 
| for($i=$days; $i<33: Sit){ // 输 出 本 月 的 天 数 
| if(!checkdate($month,$i,$year)) { // 判 断 时 间 是 否 合理 
| return($i-1); // 输 出 时 间 值 
9 
中 
| 
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应 用 示例 | 
本 示例 应 用 自 定义 函数 和 Smarty 模板 中 的 {html table} 函 数 在 Smarty 模板 页 中 生成 一 个 日 | 
历 。 其 具体 步骤 如 下 : | 
(1) 创建 index.php 文件 ， 载 入 Smarty 模板 的 配置 文件 ， 封 装 自 定义 函数 ， 将 生成 的 日 | 
历数 据 赋 给 指定 的 模板 变量 。 其 关键 代码 如 下 : | 


<2?php 
require_once("config php"); // 调 用 smarty 文件 | 
$date=date("n/dd/Y"); // 获 取 当 前 时 间 | 
$week_days=array("Sun"=>0,"Mon"=>1,"Tue"=>2,"Web"=>3,"Thu"=>4,"Fri"=>5,"Sat"=>6,); | 
/定义 星期 数组 | 
$total day of month=get total day($date); // 获 取 当 前 日 期 | 
$starting day=$week days[date("D",strtotime($date))]+1; /获取 当前 日 期 的 周 时 间 | 
foreach(array keys($week days) as $day){ /将 周 日 期 对 应 的 字符 串 存储 到 数组 中 
$days[]-$day: 
for($i=0;$i<$starting day:$it+H){ /定义 周 日 期 字符 串 
$days[]="&nbsp:"; 
b 
for($i=1; $i<($total day_ of month+1);$i++){ /对 当前 日 期 描 红 输出 
if($i=—date("d"){ 
$days[]="<font color=#FF0000>".$1."</font>"; 
}else{ 
$days[]=$i; 
} 
$smarty->assign("title"," 在 线 日 历 "); /定义 模板 变量 
$smarty->assign("special days",$days): /输出 日 历数 据 
$smarty->display("calendar.html"); // 指 定 模板 页 
function get total day($date){ // 自 定义 函数 
$time_stamp=strtotime($date): /获取 当前 时 间 的 时 间 戳 
$month ar=explode("/",$date): // 对 当前 时 间 进 行 拆 分 | 
$month=$month_ar[0]: // 将 拆 分 结果 直接 的 月 份 值 赋 给 指定 变量 | 
$year=$month ar[2]:; // 将 拆 分 结果 值 赋 给 指定 的 年 份 变量 | 
echo $year ."-". $month; // 输 出 当前 的 年 、 月 值 
$days=date("j",mktime(0,0,0,$month+1,0,$yearn)); /获取 当前 时 间 中 本 月 的 天 数 | 
for($i=$days; $1<33; SitHD{ // 输 出 本 月 的 天 数 | 
if(!checkdate($month,$i,$year)){ // 判 断 时 间 是 否 合理 | 
retun($i-1); // 输 出 时 间 值 | 
} | 
} | 
} | 
?> 


(2) 在 模板 文件 calendarhtml 中 输出 模板 变量 传递 的 值 ， 通 过 {html table} 函 数 生成 表格 
输出 当前 时 间 日 历 。 其 关键 代码 如 下 : 
<title>{$title}</title> 
</head> 
<body> 
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{html table loop=$special days cols=7 } 


</body> 
| </html> 
会 | “运行 结果 如 图 14.18 所 示 。 人 
专家 点 评 汪汪 是 二 是 大 
在 应 用 {html_table} 函 数 生成 表格 时 ,不 但 可 以 根据 指 下 本 下 本 而 
| 定 的 列 数 ， 生 成 规定 列 的 表格 ， 而 且 还 可 以 指定 表格 颜色 
| 及 表格 附加 属性 值 等 。 例 如 ， 图 14.18 Smarty 模板 中 的 页 面 设计 


<% {html table loop=$data}%> 

<% {html table loop=$data cols=4 table_attr='border="0"}%> 

<% {html table loop=$data cols=4 tr attr=$tr}%> 
| 另外 ,在 本 示例 中 将 Smarty 模板 引擎 的 配置 方法 封装 到 config php 文件 中 , 在 运行 示例 时 ， 
| 需要 配置 SMARTY PATH 常量 的 值 以 及 正确 地 指定 项 目的 存储 文件 夹 ， 和 否则 将 导致 程序 运行 
| 出 错 。 
| 在 本 示例 中 ,默认 将 示例 存储 于 Apache 服务 器 的 MR\14 文件 夹 下 ，Smarty 模板 引擎 存储 
| 位 置 是 MR\14\smarty\libs\Smarty.class.php。 


问题 271 如何 应 用 Smarty+PDO 完成 数据 的 分 页 显示 ? 


| 问题 阐述 


| Smarty 模板 引擎 实现 网 页 动静 分 离 ，PDO 是 数据 库 抽象 层 实现 对 数据 库 的 操作 ， 那 么 如 
| 何 应 用 Smarty 模板 引擎 +PDO 数据 库 抽象 层 完成 数据 库 中 数据 的 分 页 输出 呢 ? 


专家 解答 


| 应 用 Smarty+PDO 完成 数据 的 分 页 显示 ， 在 编写 分 页 代码 之 前 需要 准备 两 方面 内 容 : 一 是 
| Smarty 类 库 及 所 需 的 各 级 子 目 录 ; 二 是 创建 数据 库 表 及 插入 分 页 所 需要 的 多 条 记录 。 

| (1) 假设 程序 的 主 目录 为 4， 在 主 目录 下 创建 一 个 conn 目录 ， 将 Smarty 类 库 所 在 的 文 
| 件 夹 libs 复制 到 该 目录 中 。 在 conn 目录 中 创建 templates 目录 ， 用 来 存储 程序 的 模板 文件 。 创 
| 建 templates_c 和 cache 两 个 目录 ， 分 别 用 于 存储 编译 过 的 模板 文件 和 缓存 文件 。 

| (2) 假设 连接 的 是 localhost 主机 的 MySQL 数据库， 用户 名 和 密码 分 别 是 “root” 和 “111”， 
| 并 且 在 数据 库 “db_database” 中 创建 tb_user 数据 表 ， 然 后 通过 phpMyAdmin 工具 向 数据 表 中 
| 随意 插入 多 条 用 户 记录 ， 以 供 分 页 程序 使 用 。 

| (3) 编译 4 个 文件 ， 包 括 3 个 PHP 脚本 和 一 个 模板 文件 。 每 个 文件 的 作用 及 所 存放 的 目 
| 录 如 表 14.7 所 示 。 


表 14.7 Smarty+PDO 数据 分 页 所 用 文件 解析 


| 文件 名 称 说 明 


声明 数据 库 操作 类 、 分 页 操作 类 所 在 文件 。 该 文件 存放 在 程序 主 目录 下 的 conn 
欧 件 炎 中 


found.database.php 
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对 分 页 类 进行 实例 化 文件 。 该 文件 存放 在 程序 主 目 录 下 的 conn 文 件 夹 
主 索引 文件 , 加 载 Smarty 类 并 初始 化 Smarty 对 象 。 获取 指定 页 面 所 需要 的 数据 记 


connect.database.ph] 


mss php 录 ， 并 发 送 到 模板 中 。 该 文件 存储 在 主 目 录 下 
ee 模板 文件 , 以 HTML 表 格 形式 输出 指定 页 面 的 记录 ,调用 分 页 类 , 实现 分 页 功能 。 


该 文件 存放 在 主 目录 下 的 子 目 录 templates 中 


应 用 Smarty+PDO 实现 数据 分 页 的 具体 过 程 如 下 : 
(1) 创建 ConnDB 类 ， 定 义 ConnDB 方法 为 成 员 变量 赋值 ， 实 现 通过 PDO 连接 MySQL 
数据 库 。 代 码 如 下 : 


<2php 
/数据 库 连 接 类 
class ConnDB{ 
Var $dbtype; 
var $host: 
Var $user:; 
Var $pwd; 
var $dbname; 
// 构 造 方法 
function ConnDB($dbtype,$host, $user,$pwd, $dbname){ 
$this->dbtype=$dbtype: 
S$this->host=$host; 
S$this->user=$user; 
Sthis->pwd=$pwd: 
S$this->dbname=$dbname; 


} 
// 实 现 数据 库 的 连接 并 返回 连接 对 象 
function GetConnIdO { 
if($this->dbtype—"mysql" || $this->dbtype—"mssql"){ 
$dsn="$this->dbtype:host=$this->host:dbname=$this->dbname"; 
}else{ 
$dsn="$this->dbtype:dbname=$this->dbname"; 
) 
ty { 
$conn = new PDO($dsn, $this->user, $this->pwd); // 初 始 化 一 个 PDO 对 象 , 就 是 
创建 了 数据 库 连 接 对 象 $pdo 
S$conn->query("set names gb2312"); 
Teturm $conn; 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage0 . "<br/>"); 
由 


) 


这 就 是 ConnDB 类 ， 它 存储 于 示例 根 目录 下 conn 文件 夹 下 的 found.database.php 文件 中 。 
(2) 创建 AdminDB 类 ， 定 义 ExecSQL 方法 ， 根 据 传递 的 SQL 语句 执行 不 同 的 操作 ， 例 
如 ， 如 果 是 select， 则 执行 查询 操作 ， 返 回 查 询 结果 ; 如 果 是 insert 或 update， 则 直接 判断 是 和 否 
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执行 成 功 。 代 码 如 下 : 
/数据 库 管理 类 
class AdminDB{ 
function ExecSQL($sqlstr, $conn){ 
$sqltype=strtolower(substr(trim($sqlstr),0,6)); 
S$rs=$conn->prepare($sqlstr):; // 准 备查 询 语句 
Srs->execute(); // 执 行 查询 语句 并 返回 结果 集 
if($sqltype—"select"){ 
$array=$rs->fetchAll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 数据 
这 count($array) 一 0 || $rs 一 包 lse) 
Tetum false; 
else 
Teturn $array; 
}elseif ($sqltype=="update" || $sqltype—"insert" || $sqltype—"delete"){ 
if($rs) 
Tetum true; 
else 
return false:; 


} 
(3) 创建 SepPage 类 ， 定 义 ShowData 方法 ， 执 行 查询 语句 ， 将 查询 结果 返回 到 一 个 数组 


| 定义 ShowPage 方法 ， 创 建 分 页 超 链接 ， 返 回 值 为 超 链接 的 字符 串 。 其 关键 代码 如 下 : 


/分 页 类 
class SepPage{ 
Var $rs; 
Var $pagesize; 
Var $nowpage; 
Var $array; 
Var $conn; 
Var $sqlstr; 
function ShowData($sqlstr,$conn,$pagesize, $nowpage){ // 定 义 方法 
if(!isset($nowpage) || $nowpage—"") /判断 变量 值 是 否 为 空 
S$this->nowpage=1: // 定 义 每 页 起 始 页 
else 
S$this->nowpage=$nowpage; 
S$this->pagesize=$pagesize; // 定 义 每 页 输出 的 记录 数 
S$this->conn=$conn; // 连 接 数 据 库 返 回 的 标识 
S$this->sqlstr=$sqlstr; // 执 行 的 查询 语句 


Soffset=($this->nowpage-1)*$this->pagesize; 
$sql=$this->sqlstr." limit $offset, $this->pagesize"; 
$result=$this->conn->prepare($sql): // 准 备查 询 语句 
$result->execute(): // 执 行 查询 语句 ， 并 返回 结果 集 
S$this->array=$result->fetchAll(PDO::-FETCH_ASSOC);// 获 取 结 果 集 中 的 所 有 数据 
这 count($this->array) 一 0 || $this->array 一 false) 
Tetum false: 
else 
Teturm $this->array: 
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function ShowPage($contentname,$utits,$anothersearchstr,$anothersearchstrs,$class){ 


$str—""; 

Sres=$this->conn->prepare($this->sqlstr): // 准 备查 询 语句 

Sres->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 
$this->array=$res->fetchAll(PDO::-FETCH_ASSOC); // 获 取 结 果 集 中 的 所 有 数据 
Srecord=count($this->array): // 统 计 记 录 总 数 
S$pagecount=ceil($record/$this->pagesize); 1/ 计算 共有 几 页 


$str.=$contentname."&nbsp;".$record."&nbsp;".$utits."&nbsp; 每 页 &nbsp:".$this->pagesize. 
"&znbsp;".$utits."&nbsp; 第 &nbsp;".$this->nowpage."&nbsp; 页 / 共 &nbsp;".$pagecount."&nbsp; 页 "; 
S$str.="&nbsp;&nbsp;&nbsp;&nbsp;"; 
if($this->nowpage!=1) 
$str.="<a href=".$ SERVER[PHP SELF']."?page=l&page type=".$anothersearchstr." 
&parameter2=".$anothersearchstrs." class=".$class."> 首 页 </a>"; 
else 
$str.="<font color=#555555 人 > 首页 </font>"; 
$str.="&nbsp;"; 
if($this->nowpage!=1) 
$str.="<a 
href=".$_SERVERI['PHP SELF']."?page=".($this->nowpage-1)."&page_type=".$anothersearchstr."&parameter2=". 
$anothersearchstrs." class=".$class."> 上 一 页 </a>"; 
else 
$str.="<font color=#555555> 上 一 页 </font>"; 
$str.="&nbsp;"; 
if($this->nowpage!=$pagecount) 
$str.="<a 
href=".$_SERVERI'PHP_SELF']."?page=".($this->nowpage+1)."&page_ type=".$anothersearchstr."&parameter2=" 
.$anothersearchstrs." class=".$class."> 下 一 页 </a>"; 
else 
$str.="<font color='#555555 作 下 一 页 </font>"; 
$str.="&nbsp;"; 
if($this->nowpage!=$pagecount) 
S$str.="<a href=".$_SERVERI['PHP_SELF']."?page=".$pagecount."&page_type= 
".$anothersearchstr." &parameter2=".$anothersearchstrs." class=".$class."> 尾 页 </a>"; 
else 
$str.="<font color=#555555 作 尾 页 </font>"; 
这 count($this->array) 一 0 || $this->array 一 false) 
Tetum "无 数据 ! "; 
else 
Tetum $str; 


) 


(4) 包含 Smarty.class.php 文件 ， 创 建 SmartyProject 类 ， 定 义 SmartyProject 方法 ， 设 置 
Smarty 模板 文件 、 编 译文 件 、 缓 存 文 件 和 配置 文件 的 存储 路 径 。 


require('conn/libs/Smarty.class.php'): // 包 含 Smarty 类 文件 
class SmartyProject extends Smarty{ 
function SmartyProjectO{ 
S$this->template_dir = "conn/templates/"; /设置 模板 文件 路 径 


$this->compile dir= "conn/templates_c/": /设置 编译 文件 路 径 
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| Sthis->config dir = "conn/configs/"; /设置 配置 文件 路 径 


(5) 在 conn 文件 夹 下 的 connect.database.php 文件 中 对 分 页 类 进行 实例 化 ， 返 回 对 象 为 
| $seppage。 

(6) 在 indexphp 文件 中 调用 分 页 类 中 的 方法 ， 首 先 判断 是 否 存在 缓存 文件 ， 如 果 不 存在 ， 
| 则 执行 数据 的 分 页 查询 和 输出 ， 并 将 返回 结果 赋 给 模板 变量 ;如 果 存 在 ， 则 执行 缓存 文件 。 其 
| 代码 如 下 : 


| <2?php 

| include_once 'conn/connect.database.php'; /调用 类 中 方法 

| $arr_page=$seppage->ShowData("select * from tb_user ",$conn,5,$_GETI"page"]); /执行 分 页 查询 

| 这 !$arr page){ 

| $smarty->assign("pages","F"): 

| }else{ 

| $smarty->assign("pages","T"); 

| $smarty->assign("showpage",$seppage->ShowPage(" 用 户 "," 个 ","","","al")); 

| // 定 义 分 页 数据 模板 变量 

| $smarty->assign("arr pages",$arr page); // 将 查询 的 结果 赋 给 模板 变量 
} 

| $smarty->display('14.8/index.html'): // 指 定 模 板 页 


| (7) 在 模板 文件 index html 中 通过 section 语句 循环 输出 模板 变量 中 传递 的 数据 ， 通 过 模 
| 板 变量 输出 分 页 超 链接 。 其 代码 如 下 : 


{section name=id loop=$arr pages} 
<tr> 
<td height="20" align="center" bgcolor="#FFFFFF"> {$arr_pages[id].user}</td> 
<td align="center" bgcolor="#FFFFFF"> {$arr_pages[id].passltruncate:5:"..."}</td> 
<td align="center" bgcolor="#FFFFFF">{$arr_pages[id].dates}</td> 


| </tr> 

| {/section} 

| <tr> 

| <td height="35" colspan="3" align="center" bgcolor="#FFFFFF">{$showpage}</td> 

| </tr> 

| 运行 结果 如 图 14.19 所 示 。 EE 

| 专家 点 评 二 二 

| 在 Smarty 模板 页 中 也 可 以 柑 入 上 一 Se 

| PHP 脚本 , 其 应 用 的 是 Smarty 中 的 ee et 

| PHP 标签 。PHP 标签 中 的 语句 是 否 | 加 Er 
被 处 理 取决 于 $php_ handling 的 设 月 户 了 2 个 专 页 5 十 第 1 页 / 共 T 页 首页 上 一 页 下 一 页 展 R 
置 。 通 常情 况 下 是 不 需要 在 模板 页 本 


可 以 使 用 。 
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问题 272 
问题 273 
问题 274 
问题 275 
问题 276 
问题 277 
问题 278 
问题 279 
问题 280 
问题 281 
问题 282 


吾 吾 于 于 于 于 于 于 于 吾 至 


什么 是 ThinkPHP? 有 哪些 特点 ? 在 哪里 下 载 ? 
如 何 自动 生成 项 目 目录 ? 

ThinkPHP 的 命名 规范 有 哪些 ? 

ThinkPHP 创建 项 目的 基本 流程 是 什么 ? 

如 何 应 用 ThinkPHP 中 的 CURD 操作 用 户 信息 ? 

如 何 应 用 ThinkPHP 实现 用 户 登 录 和 数据 分 页 输出 ? 
如 何 应 用 验证 码 类 和 分 页 类 完成 用 户 登录 和 分 页 输出 ? 
如 何 通 过 ThinkPHP 中 扩展 类 生成 中 文 验证 码 ? 

如 何 实现 传递 查询 条 件 的 分 页 ? 

如 何 应 用 ThinkPHP 中 的 扩展 类 上 传 文件 ? 
ThinkPHP 框架 中 的 模板 引擎 


世 
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问题 272 ”什么 是 ThinkPHP? 有 哪些 特点 ? 在 哪里 下 载 ? 


问题 阐述 

什么 是 ThinkPHP? 有 哪些 特点 ? 在 哪里 下 载 ? 
专家 解答 

ThinkPHP 可 以 更 方便 和 快捷 地 开发 和 部 署 应 用 ， 其 不 仅仅 是 企业 级 应 用 ， 任 何 PHP 应 用 
开发 都 可 以 从 ThinkPHP 的 简单 和 快速 的 特性 中 受益 。ThinkPHP 本 身 具 有 很 多 的 原创 特性 ,并 
且 倡导 大 道 至 简 、 开 发 由 我 的 开发 理念 ， 用 最 少 的 代码 完成 更 多 的 功能 ， 其 宗旨 就 是 让 Web 
应 用 开发 更 简单 、 更 快速 。 

ThinkPHP 遵循 Apache2 开源 许可 协议 发 布 ， 意 味 着 可 以 免费 使 用 ThinkPHP， 甚 至 允许 把 
基于 ThinkPHP 开发 的 应 用 开源 或 商业 产品 发 布 /销售 。 

2. ThinkPHP 框架 的 特点 

ThinkPHP 是 一 个 性 能 卓越 并 且 功 能 丰富 的 轻 量 级 PHP 开发 框架 。 其 宗旨 就 是 让 Web 应 用 
开发 更 简单 、 更 快速 。ThinkPHP 值得 推荐 的 特性 包括 : 

加 ”类 库 导入 : ThinkPHP 是 首先 采用 基于 类 库 包 和 命名 空间 的 方式 导入 类 库 ， 让 类 库 导 
入 看 起 来 更 加 简单 清晰 ， 而 且 还 支持 冲突 检测 和 别名 导入 。 为 了 方便 项 目的 跨 平台 
移植 ， 系 统 还 可 以 严格 检查 加 载 文件 的 大 小 写 。 

回 URL 模式 : 系统 支持 普通 模式 .PATHINFO 模式 .REWRITE 模式 和 兼容 模式 的 URL 
方式 ， 支 持 不 同 的 服务 器 和 运行 模式 的 部 署 ， 配 合 URL 路 由 功能 ， 可 以 随心 所 欲 地 
构建 需要 的 URL 地 址 和 进行 SEO 优化 工作 。 

回 ”编译 机 制 : 独创 的 核心 编译 和 项 目的 动态 编译 机 制 ， 有 效 减少 OOP 开发 中 文件 加 载 
的 性 能 开销 。 

回 ”查询 语言 : 内 建 丰富 的 查询 机 制 ， 包 括 组 合 查询 、 复 合 查询 、 区 间 查 询 、 统 计 查 询 、 
定位 查询 、 动 态 查 询 和 原生 查询 ， 让 查询 数据 更 加 简洁 高 效 。 

视图 模型 轻松 动态 地 创建 数据 库 视 图 ， 多 表 查 询 不 再 烦恼 。 

回 ”分 组 模块 : 不 用 担心 大 项 目的 分 工 协调 和 部 署 问题 ， 分 组 模块 可 以 解决 跨 项 目的 难题 。 

模板 引擎 : 系统 内 建 了 一 款 卓 越 的 基于 XML 的 编译 型 模板 引擎 , 支持 两 种 类 型 的 模 
板 标签 ， 融 合 了 Smarty 和 JSP 标签 库 的 思想 ， 支 持 标签 库 扩展 。 通 过 驱动 还 可 以 支 
持 Smarty、EaseTemplate、TemplateLite 和 Smart 等 第 三 方 模板 引擎 。 

Ajax 支持 :内 置 Ajax 数据 返回 方法 ， 支 持 JSON、XML 和 EVAL 格式 返回 客户 端 ， 
并 且 系统 不 绑 定 任何 Ajax 类 库 ， 可 随意 使 用 自己 熟悉 的 Ajax 类 库 进行 操作 。 

缓存 机 制 系统 支持 包括 文件 方式 、APC、Db、Memcache、Shmop、Eaccelerator 
和 Xcache 在 内 的 多 种 动态 数据 缓存 类 型 以 及 可 定制 的 静态 缓存 规则 ,并 提供 了 快捷 
方法 进行 存 取 操作 。 

3. 环境 要 求 
ThinkPHP 可 以 支持 Windows/Unix 服务 器 环境 ， 可 运行 于 包括 Apache、IIS 在 内 的 多 种 
sd3de 
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Web 服务 器 。 需 要 PHP 5.0 及 以 上 版 本 支持 。 支 持 MySQL、MsSQL、PgSQL、Sqlite 和 Oracle | 
等 数据 库 。 | 
4. 下 载 ThinkPHP 框架 | | 
ThinkPHP 是 一 个 免费 开源 、 快 捷 、 简 单 的 OOP 轻 量 级 PHP 开发 框 染 。 它 遵循 Apache 2 | 医嘱 
开源 协议 发 布 ， 是 为 了 敏捷 的 企业 级 开发 而 诞生 的 。 获 取 ThinkPHP 的 方式 有 很 多 。 | 
官方 的 网 站 为 ，http:/thinkphp.cn。 
SVN 的 下 载 地 址 为 : | 
完整 版 本 http://thinkphp.googlecode.com/svn/trunk 
核心 版 本 http://thinkphp.googlecode.com/svn/trunk/ThinkPHP 


专家 点 评 
1. 什么 是 MVC 


MVC 是 一 种 经 典 的 程序 设计 理念 ， 此 模式 将 应 用 程序 分 为 3 个 部 分 : 模型 层 (Model)、 
视图 层 (View)、 控 制 层 (Controller)，MVC 是 这 3 个 部 分 英文 字母 的 缩写 。 


说 明 : | 
MVC 设计 模式 产生 的 原因 是 : 应 用 程序 中 用 来 完成 任务 的 代码 一 一 模型 层 (也 叫 “ 业 务 逻 | 
辑 ”),， 通常 是 程序 中 相对 稳定 的 部 分 , 重用 率 高 ; 而 与 用 户 交互 界面 一 一 视图 层 ， 却 经 常 改变 。 | 
如 果 因 需求 变动 而 不 得 不 对 业务 逻辑 代码 修改 ,或 者 要 在 不 同 的 模块 中 应 用 到 相同 的 功能 而 重 | 
复 地 编写 业务 逻辑 代码 , 不 仅 会 降低 整体 程序 开发 的 进度 , 而 且 会 使 未 来 的 维护 变 得 非常 困难 。 | 
因此 ,将 业务 逻辑 代码 与 外 观 分 离 ， 可 以 更 方便 地 根据 需求 改进 程序 ， 这 就 是 MVC 设计 模式 。 | 


在 PHP Web 开发 中 ，MVC 设计 模式 的 各 自 功能 及 相互 关系 如 图 15.1 所 示 。 


(视图 层 ) 


| 


Controller 


(控制 层 ) 


图 15.1 MVC 关系 图 


模型 层 (Model) | 
模型 层 是 应 用 程序 的 核心 部 分 , 可 以 是 一 个 实体 对 象 或 一 种 业务 罗 辑 , 它 之 所 以 称 为 模型 ，| 
是 因为 它 在 应 用 程序 中 有 更 好 的 重用 性 和 扩展 性 。 | 
视图 层 (View) | 
视图 层 提供 应 用 程序 与 用 户 之 间 的 交互 界面 ， 在 MVC 理论 之 中 ， 这 一 层 并 不 包含 任何 的 | 
业务 逻辑 ， 仅 提供 一 种 与 用 户 交 互 的 视图 。 | 
回 ”控制 层 (Controller) | 
控制 层 用 于 对 程序 中 的 请 求 进行 控制 ， 其 作用 就 像 国 家 的 宏观 调控 ， 它 可 以 选择 调用 哪些 | 
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DT 
| 视图 或 者 调用 哪些 模型。 
| 2. 什么 是 单一 入 口 
人 |。 单一 入 口 通常 是 指 一 个 项 目 或 应 用 具有 一 个 统一 《但 并 不 一 定 是 唯一 ) 的 入 口 文件 ,也 就 是 
峡 中 说 , 项 目的 所 有 功能 操作 都 是 通过 这 个 入 口 文件 进行 的 , 并 且 往往 入 口 文件 是 第 一 步 被 执行 的 。 
单一 入 口 的 好 处 是 项 目 整体 比较 规范 ， 因 为 同一 个 入 口 ， 往 往 其 不 同 操作 之 间 具 有 相同 的 
规则 。 另 外 一 个 方面 就 是 单一 入 口 控制 较为 灵活 ， 因 为 拦截 方便 ， 类 似 如 一 些 权限 控制 、 用 户 
| 登录 方面 的 判断 和 操作 可 以 统一 处 理 。 


问题 273 如何 自动 生成 项 目 目录 ? 


| 问题 阐述 

| 在 ThinkpHp 框 奥 中 如 何 自动 生成 项 目 目录 ? 

| 专家 解答 

| 下面 通过 一 个 示例 讲解 在 ThinkPHP 框架 中 如 何 自动 生成 项 目 目录 。 项 目 目录 名 称 为 16.1， 
| 其 操作 步骤 如 下 : 

(1) 在 网 站 根 目 录 下 创建 文件 夹 ， 并 命名 为 16.1。 

(2) 将 ThinkPHP 核心 类 库存 储 于 16.1 目录 下 。 

(3) 编写 入 口 文件 index.php， 将 其 存储 于 16.1 目录 下 。index.php 文件 代码 如 下 : 


pmp 
| define(THINK_ PATH', 'ThinkPHP"); /定义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
| define(CAPP_ NAME', '16.1"): // 定 义 项 目 名 称 
| define(APP_PATH'，.); /定义 项 目 路 径 
| Tequire(THINK_ PATH."/ThinkPHPphp"): /加 载 框架 入 口 文件 
| App::run(); /实例 化 一 个 网 站 应 用 实例 
! ?> 
| 在 运行 此 文件 之 前 ， 查 看 16.1 项 目的 文件 夹 架 构 ， 如 图 15.2 所 示 。 
=- 一 
| Le ETT 万 
| 扯 织 ” 包含 到 库 中 ” 共享 务 建 文件 去 = 
| 筷 本 二 (C) = 名称 
| cr A DY ThinkpHp 
| | pcs 本 地 开盘 (Ej indexphp 
[3 本 地 焉 盘 (G:] 司 
3 本 地 王 得 (HH) 
县 MSN 上 的 -我 的 网 站 ”  -* Mm 


图 15.2 项 目 文件 夹 架构 


| 在 正 浏 览 器 中 运行 此 项 目 ， 将 输出 如 图 15.3 所 示 的 运行 结果 ， 此 为 ThinkPHP 提供 的 测 
| 试 内 容 。 再 次 查看 16.1 项 目 文件 夹 ， 结 果 显 示 如 图 15.4 所 示 ， 在 项 目 根 目录 下 自动 生成 项 目 
| 目录 。 


了 。436 。 


= = 
EO" ,1 -5 户 


让 ”所 到 闫 ”二 享 ” Hea 二 ~ 团 @ 
J 广汉 x 人 x 项 
Pe 本 2011/5/23 17340 。 文 (4 
国 E 百 习 项 conf 2011/5/23 17:10 
加 二 Sia rg 2011/5/23 17:10 
ox 最 loz 四 tb 2011/5/23 17:10 
hep//192168150/MR/15/161/ - Wndows ntemet Eplerer et ee Sa 
GEO- hap//192.168.150/MR/15/151/ | 好 X x NP on ne D7” BThinkoHp 20115123 17:02 
= == 人 项 Tpl 2011/5123 1730 
dm | 六 wms | 大 ep/192163159/MR/1Sn EL: 


Dindexphp 2011/5/23 17:.02 rh 


^_ 人 ^ Hallo 欢 迎 使 用 ThinkpHp 日 


图 15.3 已 连接 到 ThinkPHP 框架 图 15.4 自动 生成 的 项 目 目录 
专家 点 评 


在 实际 开发 过 程 中 ， 目 录 结构 往往 由 于 项 目的 复杂 而 变 得 复杂 。 这 里 向 大 家 推荐 两 套 标准 | 


的 目录 部 署 方 案 : 方案 一 如 图 15.5 所 示 。 
方案 二 采用 分 组 模块 ， 如 图 15.6 所 示 。 


项 目 名 称 
目录 
已 一 项目 多 1 日 公 夫 目 录放 秆 大 目 公共 加 救 
日 局 Am 3dmin 后 台 管 理 斋 目 目录 项 目 配 置 目录 ， 阔 和 天 轩 文件 
日 Hom 需 目 目录 项 目 训 也 日 杂 (可 选 ) 
Mean. ME 项 四 时 昌 录 ， 通 荣 包 活 Action 和 Modd 目 采 


项 目 配置 目录 ， 训 | 项 目 运行 时 目录 ， 包括 Cache，Temp ，DatsfnLog 


le a 项 目 顶 板 目 录 
其 忆 邓 ， 通 党 包括 Acior 和 Nfode 上 同好 Pubic 网 站 公共 目录 
et 包 必 Cashe 、Temp、DatsfnLeg Css 样 式 文件 夫 
图 片 文件 夫 
四 Ti 贡 呆 文件 天 
admin php 网 站 的 后 台 入 口 文件 ThiskpHP 系 统 目 录 
indexphp 同 站 的 入 口 文件 index-php 网 站 的 入 口 文件 
图 15.5 项 目 部 署 方 案 (一 ) 图 15.6 项 目 部 署 方案 (二 ) 


这 样 部 署 的 好 处 是 系统 目录 和 项 目 目录 可 以 存储 于 非 Web 访问 目录 下 ,而 网 站 目录 下 只 需 | | 
放置 Public 公共 目录 和 index.php 入 口 文件 (如 果 是 多 个 项 目 ， 则 每 个 项 目的 入 口 文件 都 需要 | 


放 到 Web 目录 下 )， 这 样 就 可 以 大 大 提高 网 站 的 安全 性 。 
问题 274 ThinkPHP 的 命名 规范 有 哪些 ? 


问题 前 述 


ThinkPHP 框架 有 其 自身 的 一 定 规范 , 要 应 用 ThinkPHP 框架 开发 项 目 ， 吉 要 民明 守 它 的 | 
规范 。ThinkPHP 的 命名 规范 有 哪些 呢 ? 


专家 解答 
下 面 就 介绍 一 下 ThinkPHP 的 命名 规范 : 
类 文件 都 是 以 .class.php 为 后 级 (这 里 指 的 是 ThinkPHP 内 部 使 用 的 类 库 文件 ， 不 
代表 外 部 加 载 的 类 库 文件 ) ， 使 用 驼峰 法 命名 ， 并 且 首 字母 大 写 ， 如 DbMysql.class 
.php。 
函数 ,配置 文件 等 其 他 类 库 文件 之 外 的 一 般 是 以 php 为 后 缀 (第 三 方 引 入 的 不 做 要 求 )。 
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Re 
确保 文件 的 命名 和 调用 大 小 写 一 致 ， 是 由 于 在 类 UNIX 系统 上 ， 对 大 小 写 是 敏感 的 

(而 ThinkPHP 在 调试 模式 下 ， 即 使 在 Windows 平台 也 会 严格 检查 大 小 写 ) 。 

类 名 和 文件 名 一 臻 《包括 上 面 说 的 大 小 写 一 致 ) ， 如 UserAction 类 的 文件 命名 是 
UserAction.class.php， InfoModel 类 的 文件 名 是 InfoModel.class.php。 

函数 的 命名 使 用 小 写字 母 和 下 划 线 的 方式 ， 如 get_client ip。 
Action 控制 器 类 以 Action 为 后 级 ， 如 UserAction、InfoAction 。 
模型 类 以 Model 为 后 级 ， 如 UserModel、InfoModel。 

方法 的 命名 使 用 驼峰 法 ， 并 且 首 字母 小 写 ， 如 getUserName。 

属性 的 命名 使 用 驼峰 法 ， 并 且 首 字母 小 写 ， 如 tableName。 

以 双 下 划 线 “ ”打头 的 函数 或 方法 作为 魔法 方法 ， 如 _ call 和 _ autoload。 

常量 以 大 写字 母 和 下 划 线 命名 ， 如 HAS_ONE 和 MANY TO MANY。 

配置 参数 以 大 写字 母 和 下 划 线 命名 ， 如 HIML_CACHE_ ON。 

语言 变量 以 大 写字 母 和 下 划 线 命名 ， 如 MY_LANG; 以 下 划 线 开头 的 语言 变量 通常 
用 于 系统 语言 变量 ， 如 CLASS NOT EXIST 。 

数据 表 和 字段 采用 小 写 加 下 划 线 方式 命名 ， 如 think user 和 user name。 
专家 点 评 
在 ThinkPHP 中 有 一 个 函数 命名 的 特例 ， 就 是 单字 母 大 写 函 数 ， 这 类 函数 通常 是 某 些 操作 

的 快捷 定义 或 者 有 特殊 的 作用 ， 如 ADSL 方法 等 ， 它 们 有 着 特殊 的 含义 。 另 外 一 点 ，ThinkPHP 
默认 使 用 UTF-8 编码 ， 所 以 应 确保 程序 文件 采用 UTF-8 编码 格式 保存 ， 并 且 去 掉 BOM 信息 头 
(去 掉 BOM 头 信息 有 很 多 方式 ， 不 同 的 编辑 器 都 有 不 同 的 设置 方法 ， 也 可 以 用 工具 进行 统一 
检测 和 处 理 )。 


网 


回回 网 同 加 罗网 网 加 


问题 275 ”ThinkPHP 创建 项 目的 基本 流程 是 什么 ? 


问题 阐述 

ThinkPHP 具有 项 目 目录 自动 创建 功能 ， 因 此 构建 项 目 应 用 程序 非常 简单 ， 用 户 只 需 定义 
好 项 目的 入 口 文件 ,在 第 一 次 访问 入 口 文件 时 ， 系 统 就 会 自动 根据 在 入 口 文件 中 所 定义 的 目录 
路 径 ， 迅 速 创建 好 项 目的 相关 目录 结构 。 在 完成 项 目 目录 结构 的 创建 后 ， 接 下 来 需要 进行 哪些 
工作 才能 成 功 创建 项 目 呢 ? 
专家 解答 

ThinkPHP 创建 项 目的 基本 流程 如 图 15.7 所 示 。 

应 用 示例 

根据 上 述 讲解 的 流程 创建 一 个 名 称 为 16.2 的 项 目 ， 并 读 取 db_database 数据 库 中 的 数据 。 


其 操作 步骤 如 下 : 
(1) 创建 db_database 数据 库 和 think_user 数据 表 。 数 据 表 结构 如 图 15.8 所 示 。 
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项 目 命名 
创建 入 口 文件 


创建 控制 类 


创建 模型 类 


图 服务 器 : localhost ， 避 数据 库 : db_database 、 团 表 : think_user 
字 有 段 天 型 整理 属性 空 靳 认 额外 

int(10) 

varchar(80) = utf8_unicode_ci 

varchar(80) utf8_unicode_ci 

varchar(90) udf8_unicode_ci 


图 15.7 ThinkPHP 创建 项 目的 基本 流程 图 15.8 数据 表 结 构 


(2) 载 入 ThinkPHP 系统 文件 ， 编 辑 入 口 文件 ndexphp， 创 建 名 称 为 16.2 的 项 目 。indexphp | 
的 代码 如 下 : | 


i 
玉 别 别 别 
汕 汕 出 出 


! 
| 
| 
| 
| 
| 

auto_increment ! 
| 
| 
| 
| 
! 
| 


< | 
define(CTHINK PATH,', '../ThinkPHP/"); // 定 义 ThinkPHP 框架 路 径 | 
define(APP_NAME', '16.2"); // 定 义 项 目 名 称 和 路 径 | 
define( APP_PATH', '); /定义 项 目 名 称 和 路 径 | 
Tequire(THINK PATH."/ThinkPHP php"): // 加 载 框 架 入 口 文 件 | 
App::run0; /实例 化 一 个 网 站 应 用 实例 | 
re | 


(3) 自动 生成 的 项 目 目录 中 已 经 创建 了 一 个 空 的 项 目 配置 文件 , 位 于 项 目的 Conf 目录 下 ， 
名 称 是 config.php。 重 新 编辑 此 文件 ， 完 成 数据 库 的 配置 。config php 文件 的 代码 如 下 : | 


return array( | 
'APP DEBUG' => tme, // 开 启 调试 模式 | 
'DB_TYPE'=> 'mysql', /| 数据 库 类 型 | 
DB_HOST'=> 'localhost', /数据 库 服务 器 地 址 | 
IDB_ NAME'=>'db_database' /数据 库 名 称 | 
'DB_USER'=>'root, /| 数据 库 用 户 名 | 
DB PWD'=>"111', /| 数据 库 密码 | 
DB_PORT'=>'3306', /| 数据 库 端口 | 
'DB_PREFIX'=>'think ', /| 数据 表 前 级 | 


(4) 在 项 目的 Lib\Action 目录 下 定位 到 自动 生成 的 IndexAction.class.php 文件 ， 这 是 
ThinkPHP 的 控制 器 , 即 Index 模块 。 重新 编辑 控制 器 的 index 方法 , 查询 指定 数据 表 中 的 数据 ， 
并 且 完 成 数据 的 循环 输出 。 其 代码 如 下 : 

<2php 
class IndexAction extends Action{ 
站 
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| 完成 数据 库 


} 


De 


$db = new Model(user): 


人 ip 六 预 知道 的 300 个 问题 


$select = $db->select(); // 查 询 数 据 
S$this->assign('select', $select): // 模 板 变量 赋值 
$this->displayO: // 输 出 模板 


<!-- 循 环 输出 查询 结果 数据 集 --> 
<volist name='select id='"user' > 
ID: {$user.id}<br/> 


用 户 名 : 


地 址 : 


{$user.user}<br/> 
{$user.address}<hr> 
</volist> 


/实例 化 模型 类 ,参数 数据 表 名 称 ， 不 包含 前 级 


(5) 在 项 目的 Tpl\default 目录 下 创建 mdex 目录 ,存储 Index 模块 的 模板 文件 index.html， 
数据 的 循环 输出 。 其 代码 如 下 : 


(6) 在 正 浏 览 器 中 输入 http:/127.0.0.1/ mr/16/example/16.2/， 其 运行 结果 如 图 15.9 所 示 。 


轿 ThinkpHp 开 和 县 - Windows Internet Explorer [El | 


GO nomusy -elslxllPe 5 


次 ws | 革 |-] 硬 ThinkpHpFRRB 。 x | 心 192168159 /locahost -| 
ID: 1 


用 户 名 : mr 
地 址 :长春 市 


ID: 2 
用 户 名 : mrsoft 
地 址 : 四 平市 


|| Process: 0.039s ( Load:0.012s Init0.007s Exec:0.019s Template:0.001s ) | DB :2 


queries 0 writes | UseMem:589 kb 


16116: 
tehe0573c70486023961c53E714aa7Tbd87 php 


和 GET 
通信 协议 :HTTPILL 


图 15.9 了 解 ThinkPHP 项 目 构建 流程 


问题 276 如何 应 用 ThinkPHP 中 的 CURD 操作 用 户 信息 ? 


| 问题 阐述 


| 如 何 应 用 ThinkPHP 中 的 CURD 操作 ， 实 现 对 用 户 信息 的 查询 、 更 新 和 删除 操作 ? 
| 专家 解答 
| CURD 是 数据 库 操作 的 缩写 词 ， 也 是 几 种 数据 库 操作 技术 的 缩写 ,“C” 代 表 创 建 (Create)， 
| “U” 代 表 更 新 (Updatej,“R” 代 表 读 取 (Read),“D” 代 表 删 除 Delete)。CURD 定义 了 用 于 处 
| 理 数据 的 基本 操作 。 

| CURD 在 具体 的 应 用 中 并 非 一 定 使 用 create、update、read 和 delete 字样 的 方法 , 但 是 它们 


完成 的 功能 是 一 致 的 。 例 如 ，ThinkPHP 就 是 使 用 add、save、select 和 delete 方法 表示 模型 的 
| CURD 操作 。 
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技术 如 下 : 


(1) 通过 M 快捷 方法 实例 化 基础 模型 类 。 在 未 定义 任何 模型 时 ， 可 以 使 用 下 面 的 方法 实 
例 化 一 个 模型 类 来 进行 操作 : | 
$User = new Model('User’); 
S$User->select():; // 进 行 其 他 的 数据 操作 
或 者 使 用 M 快捷 方法 进行 实例 化 ， 其 效果 是 相同 的 。 
$User = M('User’); 
S$User->select(); // 进 行 其 他 的 数据 操作 | 
这 种 方法 简单 高 效 ， 因 为 不 需要 定义 任何 的 模型 类 ， 所 以 支持 跨 项 目 调 用 。 其 缺点 是 因为 | 
没有 自 定义 的 模型 类 ， 因 此 无 法 写 入 相关 的 业务 逻辑 ， 只 能 完成 基本 的 CURD 操作 。 | 
(2) 连贯 操作 与 CURD 操作 配合 应 用 ， 完 成 数据 的 创建 、 更 新 、 读 取 和 删除 。 
Where: 用 于 查询 或 更 新 条 件 的 定义 。 参 数 支持 字符 串 、 数 组 和 对 象 。 | 
Order: 对 结果 进行 排序 。 排 序 方法 支持 对 多 个 字段 的 排序 ， 如 order('status descid | 
asc)。order 方法 的 参数 支持 字符 串 和 数组 ， 数 组 的 用 法 如 下 : order(array('status=> | 
"descuvid))。 | 
加 ”Limit: 结果 限制 ,在 ThinkPHP 中 ,无论 操作 的 是 MySQL、MS SQL Server 还 是 Oracle | 
数据 库 ， 其 limit 的 方法 都 是 统一 的 ， 即 limit('offset,length")。 如 limit(1,10)， 表 示 获 | 
取 从 第 一 条 记录 开始 的 10 条 记录 。 | 
回 ”查询 select() 方 法 的 返回 值 是 一 个 二 维 数 组 ， 如 果 没 有 查询 到 任何 结果 ， 则 返回 一 | 
个 空 的 数组 。 配 合 上 面 提 到 的 连贯 操作 方法 可 以 完成 复杂 的 数据 查询 。 | 
回 ”更 新 : save() 方 法 在 执行 更 新 数据 的 操作 时 ， 如 果 没 有 设置 任何 更 新 条 件 ， 且 数据 对 | 
象 本 身 也 不 包含 主键 字段 ， 那 么 该 方法 不 会 更 新 任何 数据 库 的 记录 。 | 
删除 : delete() 方 法 可 以 用 于 删除 单个 或 多 个 数据 , 主要 取决 于 删除 条 件 , 也 就 是 where | 
方法 的 参数 ， 也 可 以 用 order 和 limit 方法 来 限制 要 删除 的 个 数 ， 例 如 ， 删 除 所 有 状 | 
态 为 0 的 5 个 用 户 数据 按照 创建 时 间 排 序 。 代 码 如 下 : | 
S$User = M("User"); /实例 化 User 对 象 
$User->where('status=0)->order('create time)->limit(S)->delete0; | 
(3) 在 控制 器 中 通过 assign() 方 法 为 模板 变量 赋值 ;通过 displayO 指 定 模板 页 ， 通 过 | 
redirect() 方 法 完成 页 面 的 重 定向 。 | 
通过 assign() 方 法 将 控制 器 中 获取 的 数据 赋 给 模板 变量 ， 例 如 $this->assign(name'，| 
$value):。 | 
通过 display() 方 法 指定 模板 页 (display( 操 作 名 )) 。 调 用 当前 模块 的 其 他 操作 模板 。 | 
例如 ， 当 前 是 User 模块 下 面 的 read 操作 ， 而 要 调用 User 模块 的 edit 操作 模版 ， | 
S$this->display(edit): 。 | 
通过 redirect0 方 法 实现 页 面 的 重 定向 。 定 义 规则 分 为 如 下 3 种 : 第 一 种 基本 规则 指定 
项 目 和 模块 ， 第 二 种 数组 传 参数 ， 第 三 种 不 指定 项 目 、 模 块 ， 即 表示 当前 项 目 和 模块 。 | 
redirect ('[ 项 目 ://][ 路 由 @][ 分 组 名 -模块 /| 操作 ? 参数 1= 值 1[& 参 数 N= 值 N]) | 
redirect ('[ 项 目 ://][ 路 由 @][ 分 组 名 -模块 /| 操作 ',array( 参 数 1=>' 值 1 [参数 N=>' 值 N])) | 
$this->redirect(Index/index',", 3. 页面 跳 转 中 /停留 5 秒 , 跳 到 Index 模块 index 操作 ， 显 示 | 
页 面 跳 转 字样 | 
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(4) 应 用 内 置 模板 引擎 (ThinkTemplate) 中 的 foreach 标签 在 模板 页 中 循环 输出 模板 变量 
传递 的 数据 。foreach 标签 的 语法 如 下 : 
<foreach name="list" item="vo" > 
{$vo.1d} 
{$vo.name} 
</foreach> 
foreach 标签 用 于 循环 输出 ， 它 比 volist 标签 简洁 ,但 没有 volist 标签 那么 多 的 功能 。 另外， 
foreach 标签 可 以 对 对 象 进行 遍历 输出 ， 而 volist 标签 通常 是 用 于 输出 数组 。 
应 用 示例 
应 用 ThinkPHP 中 的 CURD 操作 实现 对 用 户 信息 的 查询 、 更 新 和 删除 操作 。 其 具体 步骤 如 下 : 
(1) 创建 253 项 目 根 目录 ， 在 根 目 录 下 创建 项 目 文件 夹 App 和 Public 文件 夹 存储 CSS、 
图 片 和 JS 脚本 等 文件 。 
(2) 在 253 项 目 根 目录 下 编辑 index.php 入 口 文件 。 其 关键 代码 如 下 : 


<2php 

define(CTHINK PATH',' /ThinkPHP)): /定义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
define(APP NAME', 'App)); /定义 项 目 名 称 

define(APP PATH',' /App)); /定义 项 目 路 径 

Iequire(THINK_ PATH."/ThinkPHPphp"): /加 载 框架 入 口 文件 

App::run0; /实例 化 一 个 网 站 应 用 实例 

?> 


(3) 在 正 浏 览 器 中 运行 入 口 文件 ， 自 动 生成 项 目 目录 。 
(4) 定位 到 App\Conf 目录 下 ， 编 辑 config.php 文件 ， 并 通过 PDO 连接 MySQL 数据 库 。 
其 代码 如 下 : 


<?php 
Tetum array( 
'DB_TYPE'=> 'pdo', 
/注意 DSN 的 配置 针对 不 同 的 数据 库 有 所 区 别 
'DB_DSN'=> ‘mysql:host=localhost:dbname=db_database', 
DB_USER'=>'root', 
DB PWD'=>'111', 
'DB_PREFIX'=>'think ', 
// 其 他 项 目 配 置 参数 …… 
'APP DEBUG' => false, // 关 闭 调试 模式 
有 


?> 


(5) 定位 到 App\LibWAction\ 目 录 下 , 编写 项 目 控制 器 。 创建 mdex 模块 ， 继 承 系统 的 Action 
基础 类 ， 定 义 index0 方 法 ， 以 记录 的 ID 值 为 条 件 ， 降 窜 循 环 输出 10 条 记录 。 其 代码 如 下 : 

<2php 

header("Content-Type:text/html; charset=utf-8"); 。 // 设 置 页 面 编 码 格式 

class IndexAction extends Action{ 

public function indexO{ 

$db =MCUser); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 ， 不 包含 前 级 
S$select = $db->order('id desc)->limit(10)->selectO: 
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Sthis->assign('select', $select); // 模 板 变量 赋值 

S$this->display0: /指定 模板 页 
出 | 
定义 update() 方 法 ， 首 先 根 据 超 链 接 传递 ID 值 执行 查询 ， 查 询 出 指定 的 数据 ， 并 且 将 查询 | 
结果 赋 给 指定 的 模板 变量 。 然 后 ， 判 断 表单 提交 的 邢 值 是 否 存在 ， 如 果 存 在 ， 则 以 琴 为 条 件 | 
对 指定 的 数据 进行 更 新 操作 。 其 关键 代码 如 下 : 


public function updateO{ | 
$db =MCUser); /实例 化 模型 类 ， 参 数 数据 表 名 称 ， 不 包含 前 绥 | 
S$select = $db->where(id=".$_GET['id'])->select|; | 
S$this->assign('select',$select); // 模 板 变 量 赋值 | 
S$this->display(update); /指定 模板 页 


iisset($_ POST[id])){ 
$data["user'] =$_ POST[user]; /要 修改 的 数据 对 象 属性 赋值 
$data['pass] = md5($_ POST[pass']): 
$data['address'] =$ POST['address']; 
$result=$db->where("id=".$_POST['id'])->save($data); // 根 据 条 件 保存 修改 的 数据 


这 Sresulb{ | 
$this->redirect(Tndex/index',", 2,' 数 据 更 新 成 功 ); /页面 重 定向 | 
} | 
) 
} 
定义 delete() 方 法 ， 根 据 超 链接 传递 的 ID 值 删除 数据 库 中 指定 的 记录 。 其 关键 代码 如 下 : | 
public function deleteO{ | 
$db = M('User'): // 实 例 化 模型 类 ， 参 数 数据 表 | 
名 称 ， 不 包含 前 级 | 
Sresult=$db->where('id=".$_GET['id"])->delete(); /删除 了 为 5 的 用 户 数据 | 
if($result){ | 
$this->redirect(Tndex/index',", 2. 数 据 删除 成 功 ); /页 面 重 定向 | 
| 
} | 
} 
?> 


(6) 定位 到 App\Tpldefault 目录 下 ， 创 建 mdex 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 | 
index html， 应 用 ThinkPHP 内 置 模板 引擎 中 的 foreach 标签 循环 输出 模板 变量 传递 的 数据 ; 创 | 
建 更 新 和 删除 超 链接 ， 将 指定 记录 的 ID 作为 参数 进行 传递 。 其 关键 代码 如 下 ; 


<foreach name='select item='user > ! 
<tr class="content"> | 
<td bgcolor="#FFFFFF">&nbsp; {$user.id}</td> | 
<td bgcolor="#FFFFFF">&nbsp; {$user.user} </td> ! 
<td bgcolor="#FFFFFF">&nbsp: {$user.address} </td> 
<td bgcolor="#FFFFFF"><a href=" URL /update?id={$userid}"> 更 新 </a>/<a href=" | 
URL __/delete?id={$user.id} "> 删除 </a></td> 
</tr> 
</foreach> 


(7) 在 Index 模块 文件 夹 下 ， 编 辑 update html 模板 文件 ， 创 建 表单 ， 将 从 模板 变量 中 读 
. 443 。 
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| 取 的 数据 作为 表单 元 素 的 默认 值 进行 输出 ， 将 表单 中 数据 提交 到 控制 器 的 update0 方 法 中 完成 
| 数据 的 更 新 操作 。 其 关键 代码 如 下 : 

| <form id="form?2" name="form?2" method="post" action=" URL /pdate"> 

| <table width="405" border="1" cellpadding="1" cellspacing="1" bgcolor="#99CC33" bordercolor= 


"#FFFFFF"> 
<foreach name='select' item='user > 
<tr class="content"> 
| <td bgcolor="#FFFFFF" class="right" width="103"> 名 称 : </td> 
| <td bgcolor="#FFFFFF" width="289"> <input type="hidden" name="id" id="hiddenField" 
| value=" {$userid}" /><input name="user" type="text" id="user" size="20" value=" {$user.user}" /></td> 
< 
| <tr class="content"> 
| <td bgcolor="#FFFFFF" class="right"> 密 码 : </td> 
| <td bgcolor="#FFFFFF"><input name="pass" type="password" id="pass" size="20" value= 
| "{$user.pass}" /> 
| </td> 
| </tr> 
| <tr class="content"> 
| <td bgcolor="#FFFFFF" class='right">&nbsp: 地 址 : </td> 
| <td bgcolor="#FFFFFF">&nbsp; 
| <input name="address" type="text" id="address" size="30" value="{$useraddress}" /> 
| </td> 
| </tr> 
| <tr class="content"> 
| <td bgcolor="#FFFFFF"><input type="submit" name="button" id="button" value=" 更 新 " /></td> 
| </tr> 
! </foreach> 
| </table> 
! </form> 
| 运行 结果 如 图 15.10 所 示 。 
| 据 作 
| 更 新 他 
上 用 户 信息 
| : | 明日 科技 
. TT 
| EE 
| | 握 作 
上 | 重新 全 迭 
| ET 
| EE 
| Er 
! ET 3 
| EE 
| 图 15.10 数据 更 新 和 删除 
专家 点 评 
| 在 ThinkPHP 的 模板 文件 中 有 一 种 特殊 的 字符 串 ， 可 以 对 模板 中 的 目录 进行 替换 ， 使 模板 
| 文件 的 定义 更 加 方便 ， 其 默认 的 替换 规则 如 表 15.1 所 示 。 
] .444 。 
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表 15.1 模板 中 特殊 字符 串 的 蔡 换 规则 
替换 描述 

被 蔡 换 成 当前 项 目的 公共 模板 目录 。 通 常 是 : /项 目 目 录 /Tpl/default/Public/ 
被 替换 成 当前 网 站 的 公共 目录 。 通 常 是 /Public/ 
替换 成 项 目的 模板 目录 。 通 常 是 ， /项目 目录 /Tpl/default/ 
会 替换 成 当前 网 站 的 地 址 不 含 域名 ) 
替换 成 当前 项 目的 URL 地 址 (不 含 域名 ) 
替换 成 当前 模块 的 URL 地 址 (不 含 域名 ) 
替换 成 当前 操作 的 URL 地 址 (不 含 域名 ) 
蔡 换 成 当前 的 页 面 URL 


这 些 特 殊 的 字符 串 是 严格 区 别 大 小 写 的 ,并 且 这 些 特殊 字符 串 的 替换 规则 是 可 以 更 改 或 增 
加 的 。 只 要 在 项 目 配置 文件 中 配置 TMPL PARSE STRING 即 可 完成 。 如 果 有 相同 的 数组 索引 ， 
就 会 更 改 系统 的 默认 规则 。 例 如 : 


TMPL PARSE STRING =>array( 
'_ PUBLIC _'=>'/Common', // 更 改 默 认 的 _PUBLIC_ 替换 规则 
'_ UPLOAD _'=>'/Public/Uploads/", // 增 加 新 的 上 传 路 径 蔡 换 规则 


问题 277 ”如 何 应 用 ThinkPHP 实现 用 户 登 录 和 
数据 分 页 输出 ? 


问题 曾 述 
如 何 应 用 ThinkPHP 实现 用 户 登 录 和 数据 分 页 输出 ? 
专家 解答 
要 实现 用 户 登录 和 数据 分 页 输出 主要 应 用 的 技术 有 两 方面 : 第 一 个 方面 是 通过 ThinkPHP 
框架 中 的 分 页 类 以 及 连贯 操作 Page0 方 法 实现 数据 的 分 页 输出 ; 第 二 个 方面 是 应 用 
{$Think.session.session_ idlmds} 标 签 输出 SESSION 变量 。 
(1) 通过 import("ORGUtil.Page"): 载 入 分 页 类 。 
import("ORG:Util Page"); // 导 入 分 页 类 
(2) 实例 化 分 页 类 ， 同 时 可 以 传递 3 个 参数 : 第 一 个 是 页 面 总 记录 数 ， 第 二 个 是 每 页 显 
示 的 记录 数 ， 第 三 个 为 可 选 参数 ， 通 过 分 页 超 链接 的 值 。 
$count = $db->where(address=…" 长 春 市 ")->count0: /查询 满足 要 求 的 总 记录 数 


$Page = new Page($count,1): /实例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 
(3) 调用 分 页 类 中 的 show0 方 法 ， 输 出 分 页 超 链接 。 
$show = $Page->show(); // 分 页 显示 输出 


(4) Page0: 查询 分 页 。 属 于 新 增 特性 ， 可 以 更 加 快速 地 进行 分 页 查询 。Page0 方 法 的 用 
.445 。 
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| 法 和 limit 方法 类 似 ， 格 式 为 
| Page(page[,listRows]) 
, | page 表示 当前 的 页 数 ，listRows 表示 每 页 显示 的 记录 数 。 例 如 Page(2.10)， 表 示 每 页 显示 
| 10 条 记录 ， 获 取 第 2 页 的 数据 。 
| isows 如 果 不 写 ， 则 会 读 取 limit(length') 的 值 ， 例 如 limit(25)->page(3):， 表 示 每 页 显示 
25 条 记录 ， 获 取 第 3 页 的 数据 。 如 果 limit 也 未 进行 设置 ， 则 默认 为 每 页 显示 20 条 记录 。 

应 用 示例 

| 本 示例 实现 用 户 登 录 功 能 ， 将 登录 用 户 的 信息 存储 到 SESSION 变量 中 ， 应 用 ThinkPHP 

| 中 提供 的 分 页 扩展 类 和 Page 方法 完成 数据 的 分 页 输出 。 其 具体 步骤 如 下 : 
(1) 创建 002 项 目 根 目录 ， 在 根 目录 下 创建 项 目 文件 夹 App 和 Public 文件 夹 存储 CSS、 


| 图 片 和 JS 脚本 等 文件 。 
| (2) 在 002 项 目 根 目录 下 编辑 index.php 入 口 文件 。 其 关键 代码 如 下 : 
<2php 
define('THINK PATH','/ThinkPHP)): /定义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 
define(CAPP NAME','App); /定义 项 目 名 称 
define('APP_PATH', '/App’): // 定 义 项 目 路 径 
require(THINK_ PATH."/ThinkPHPphp"): // 加 载 框架 入 口 文 件 
App::run(); /实例 化 一 个 网 站 应 用 实例 
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(3) 在 正 浏 览 器 中 运行 入 口 文件 ， 自 动 生成 项 目 目录 。 
(4) 定位 到 App\Conf 目录 下 ,编辑 config.php 文件 ， 通 过 PDO 连接 MySQL 数据 库 。 其 


| 代码 如 下 : 

<2php 

Tetum array( 
'DB_TYPE'=> 'pdo', 
/注意 DSN 的 配置 针对 不 同 的 数据 库 有 所 区 别 
'DB_DSN'=> 'mysql:host=localhost:dbname=db_database', 
'DB_USER'=>'root', 
'DB_PWD'=>'111', 
'DB_PREFIX'=>'think ', 
// 其 他 项 目 配 置 参数 ……… 
'APP DEBUG' => false, // 关 闭 调试 模式 


| (5) 定位 到 \App\Lib\Action\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 Index 模块 ， 继 承 系统 的 
| Action 基础 类 ， 定 义 index( 方 法 。 验 证 用 户 提交 的 用 户 名 和 密码 是 否 正确 ， 如 果 正 确 ， 则 将 登 
| 录用 户 名 存储 到 SESSION 变量 中 ， 并 且 将 网 页 重 定向 到 main html 页 面 。 其 代码 如 下 : 


<?php 
session start(); // 初 始 化 SESSION 变量 
header("Content-Type:text/html: charset=utf-8"); 。“” /设置 页 面 编码 格式 
class IndexAction extends Action{ 

public function indexO{ 


| iflisset($_POST['user]){ 
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iftisset($ POST['user']) && isset($ POST['pass']){ 
$db = MO; /实例 化 模型 类 ， 参 数 数据 表 名 称 ， 不 包含 前 绥 
$select = $db->query("select * from think user where user=".$ POST[user]." 
and pass=".$ POST['pass].""); // 执 行 查询 语句 ， 验 证 用 户 名 和 密码 是 否 正 确 
if($select){ 
$_ SESSION['admin]=$ POST[user]; // 将 登录 用 户 名 存储 到 SESSION 中 
Sthis->redirect(TIndex/main'", 2. 用 户 'S_POST[user]， 登录 成 功 ! 小 
// 页 面 重 定向 


Jelse{ 
S$this->redirect('Index/index',", 2, 用 户 名 或 者 密码 不 正确 ! ); // 页 面 重 定向 
} 
}else{ 
$this->redirect(Tndex/index',", 2, 用 户 名 、 密 码 不 能 为 空 ! 人 // 页 面 重 定向 
} 
b 
Sthis->displayO; 


} 
定义 main() 方 法 ， 载 入 分 页 类 ， 完 成 数据 库 中 数据 的 分 页 查询 ， 并 且 将 查询 结果 赋 给 模板 
变量 。 其 代码 如 下 : 


public function mainO{ 
S$db = M('User'): // 实 例 化 模型 类 ， 参 数 数据 表 名 称 ， 不 包含 前 级 
// 进 行 分 页 数据 查询 。 注意 page 方法 的 参数 的 前 面部 分 是 当前 的 页 数 使 用 $_GET[p] 获 取 
ifisset($_GET[p])){ /判断 分 页 变量 是 否 存在 
S$p=$_GET['p']:; 
}else{ 
$p=1; 


由 
$list = $db->where('address="" 长 春 市 ")->order('id desc)->page($p.,,1)->select0:/ 查 询 数据 
$this->assign('select',$list); /赋值 数据 集 
import("ORG Util.Page"): // 导 入 分 页 类 
$count = $db->where('address="." 长 春 市 ")->countO; // 查 询 满足 要 求 的 总 记录 数 
$Page = new Page($count,1); /实例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 
$show = $Page->show(); // 分 页 显示 输出 
$this->assign('page',$show); /赋值 分 页 输出 
S$this->display(main); /| 输出 模板 

} 


定义 validatorcode() 方 法 , 应 用 GD 库 中 的 函数 , 根据 超 链接 传递 的 值 生成 用 户 登 录 的 验证 
码 。 其 代码 如 下 : 


public function validatorcodeO{ 
header('content-type:image/png’): // 定 义 标题 png 格式 图 像 
S$im = imagecreate(65, 25): /定义 画布 
imagefill($im, 0, 0, imagecolorallocate($im, 200, 200, 200)); 。 “” // 区 域 填充 
SvalidatorCode = $_GET['code']:; // 获 取 提 交 的 值 


imagestring($im, rand(3, 5), 10, 3, substr($validatorCode, 0, 1), imagecolorallocate($im, 0, 
rand(0, 255), rand(0, 255))); 

imagestring($im, rand(3, 5), 25, 6, substr($validatorCode, 1, 1), imagecolorallocate($im, rand(0, 
255), 0, rand(0, 255))); 

imagestring($im, rand(3, 5), 36, 9, substr($validatorCode, 2, 1), imagecolorallocate($im, rand(0, 
255), rand(0, 255), 0)): 
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| KK 
KC% 
| imagestring($im, rand(3, 5), 48, 12, substr($validatorCode, 3, 1), imagecolorallocate($im, 0, 
| rand(0, 255), rand(0, 255))); 

| imagepng($im); // 生 成 PNG 图 像 

| imagedestroy(): // 销 毁 图 像 


(6) 定位 到 App\Tpl\default 目录 下 ， 创 建 mdex 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 
index .html， 载 入 CSS 样式 文件 和 JavaScript 文件 ， 创 建 表单 ， 完 成 用 户 登录 信息 的 提交 操作 。 


| 其 关键 代码 如 下 : 


<link href=" ROOT_ /Public/Css/style.css" rel="stylesheet" type="text/css" /> 
<js href=" ROOT_ /Public/Js/check.js" /> 
<form name="form1" method="post"” action=" URL /index" onSubmit="return chkinput(this)" > 
<table width="265" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td class="title" id="td"> 用 户 名 : </td> 
<td><input name="user" type="text" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 密 码 : </td> 
<td><input name="pass" type="password" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 验 证 码 : </td> 
<td> 
<input type="text" name="validatorCode" size="10" /> 
<input type="hidden" name="defValidatorCode" value="" /> 
<script language="javascript"> 
var numl=Math round(Math randomO*10000000): // 生 成 随机 数 
var num=numl.toStringO.substr(0.4): /截取 随机 数 的 前 4 个 字符 
document write("<img name=codeimg sre='" URL /validatorcode?code="+num+">"); 


/将 截取 值 传递 到 图 像 处 理 页 中 


forml.defValidatorCode value=num: // 将 截取 值 赋 给 表单 中 的 隐藏 域 
function reCodeO{ // 定 义 方法 ， 重 新 生成 验证 码 
var num1=Mathround(Math.random(Q*10000000); /生成 随机 数 
var num=numl toString(.substr(0.4): // 截 取 随 机 数 


document.codeimg.sre=" URL /validatorcode?code="+num:; 

| /将 截取 值 传递 到 图 像 处 理 页 中 
| forml.defvalidatorCode value=num: /将 截取 值 赋 给 表单 中 的 隐藏 域 
</script> 

<a href="javascript:reCode(" class="content"> 看 不 清 </a> 

</td> 

| </t> 
</table> 
| <input type="image" name="imageField" id="imageField" sre=" ROOT /Publioimages/66 05.gif' /> 
| </form> 
| (7) 在 Index 模块 文件 夹 下 编辑 main.html 文件 ， 通 过 模板 引擎 中 的 session 标签 输出 当 
前 登录 的 用 户 名 ， 然 后 通过 foreach 标签 循环 输出 模板 变量 传递 的 数据 ， 最 后 输出 模板 变量 传 
递 的 分 页 超 链 接 。 其 关键 代码 如 下 : 
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<table width="405" border="1" cellpadding="1" cellspacing="1" bgcolor="#99CC33" bordercolor= 
"#FFFFFF"> 
<tr> 
<td colspan="3" bgcolor="#FFFFFF" class="title" align="center"> 当 前 登录 用 户 : {$Think.session | 
.admin} </td> | 
</t> 
<foreach name='select item=user > 
<tr class="content"> 
<td bgcolor="#FFFFFF">&nbsp: {$user.id} </td> | 
<td bgcolor="#FFFFFF">&nbsp; {$user.user} </td> | 
<td bgcolor="#FFFFFF">&nbsp; {$user.address} </td> | 
</tr> | 
</foreach> | 


<tr class="content"> 
<td colspan="3" bgcolor="#FFFFFF">&nbsp;{$page}</td> 
</tr> 
</table> | 


运行 结果 如 图 15.11 所 示 。 | 
| 
| 大生 市 


4 各 记录 1 页 下 = 区 12 卫 二 


图 15.11 用 户 登 录 和 数据 的 分 页 输出 


435 57a 
当前 要 妇 用 户 : mr 
而 户 信息 


| 有 和 B22] 


在 利用 Page( 方 法 进行 数据 的 分 页 查询 时 ，Page( 方 法 的 第 一 个 参数 是 当前 的 页 数 ， 需 要 
使 用 $_GET[p]， 第 二 个 参数 是 当前 页 显示 的 记录 数 。 | 


问题 278 ”如 何 应 用 验证 码 类 和 分 页 类 完成 用 广 
登录 和 分 页 输出 ? 


问题 前 述 
如 何 应 用 验证 码 类 和 分 页 类 完成 用 户 登 录 和 分 页 输出 ? | 
专家 解答 


在 验证 码 类 中 应 用 buildImageVerify 0 生成 验证 码 。buildImageVerify () 方 法 的 语法 如 下 : 
buildImage Verify($length,$mode, $type,$width,$height, $verifyName) 
buildImageVerify() 方 法 的 参数 说 明 如 表 15.2 所 示 。 
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表 15.2 ”验证 码 类 中 buildImageVerify() 的 参数 说 明 
参数 说 明 
length 验证 码 的 长 度 ， 默 认为 4 位 数 
验证 字符 串 的 类 型 ， 默 认为 数字 ， 其 他 支持 类 型 有 0 字母 ，1 数 字 ; 2 大 写字 母 ，3 小 写字 


We 母 ， 4 中文， 5 混合 (去掉 了 容易 混淆 的 字符 o0LI 和 数字 01) 
type 验证 码 的 图 片 类 型 ， 默 认为 png 
width 验证 码 的 宽度 ， 默 认 会 自动 根据 验证 码 长 度 自动 计算 


height 验证 码 的 高 度 ， 默 认为 22 
verifyName | 验证 码 的 SESSION 记录 名 称 ， 默 认为 verify 


生成 验证 码 之 后 ， 需 要 在 模板 页 中 通过 <img src=" APP_ /Index/verify/" 这 输出 生成 的 验 
证 码 图 像 ， 在 控制 器 中 判断 验证 码 是 否 正 确 的 代码 如 下 : 
这 $_SESSION[verify'] (= ms POST['verify"])) { 
$this->error(' 验 证 码 错误 ! "); 
) 
应 用 示例 
本 示例 以 用 户 登 录 和 数据 的 输出 为 背景 ， 应 用 ThinkPHP 中 提供 的 验证 码 类 和 分 页 类 生成 
验证 码 ， 完 成 数据 的 分 页 输出 。 其 关键 步骤 如 下 : 

(1) 定位 到 \App\Lib\Action\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 Index 模块 ， 继 承 系统 的 
Action 基础 类 ， 定 义 index() 方 法 ， 验 证 SESSION 变量 存储 的 验证 码 与 用 户 提交 的 验证 码 是 否 
相同 。 验 证 用 户 提 交 的 用 户 名 和 密码 是 否 正 确 ， 如 果 正 确 ， 则 将 登录 用 户 名 存储 到 SESSION 
变量 中 ， 并 且 将 网 页 重 定向 到 main.html 页 面 。 其 代码 如 下 : 

<2php 
session start(); 
header("Content-Type:text/html; charset=utf-8"); 。“ // 设 置 页 面 编码 格式 
class IndexAction extends Action{ 
public function indexO{ 
iisset($_ POST[user])){ 
ifisset($_ POST[user]) && isset($ POST[pass]) &&c isset($_ POST[validatorCode])){ 
这 $_SESSION[verify] 一 md5($_ POST[validatorCode])) { 


/验证 验证 码 是 否 正确 
$db = MO; // 实 例 化 模型 类 , 参数 数据 表 名 称 , 不 包含 前 织 
S$select = $db->query("select * from think user where user=".$_ POST['user’]." 
and pass=".$_POST['pass']."™"); // 执 行 查询 语句 ， 验 证 用 户 名 和 密码 是 否 正确 


if($select){ 
$_SESSION['admin]-$_POST['user]: 
S$this->redirect(TIndex/main',"，2,' 用 户 '$_ POST[user]' 登录 成 功 ! ); 


// 页 面 重 定向 
Jelse{ 
Sthis->redirect(Index/index'", 2. 用 户 名 或 者 密码 不 正确 ! 小 
// 页 面 重 定向 
} 
Jelse{ 


Sthis->redirect(Index/index',", 2, 验 证 码 不 正确 ! );  ”// 页 面 重 定向 
} 
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}else{ 
S$this->redirect(Index/index',", 2. 用 户 名 、 密 码 不 能 为 空 ! ");// 页 面 重 定向 


上 
$this->displayO; 
} 


定义 main() 方 法 ， 载 入 分 页 类 ， 完 成 数据 库 中 数据 的 分 页 查询 ， 并 且 将 查询 结果 赋 给 模板 


变量 。 这 里 应 用 的 是 Page 类 和 Limit 方法 完成 数据 的 分 页 输出 ， 其 代码 如 下 : 


public function mainO{ 
$db = M('User'): // 实 例 化 模型 类 ， 参 数 数据 表 名 称 ， 不 包含 前 级 
import("ORG:UtilPage"); /导入 分 页 类 
$count = $db->count|; /统计 总 记录 数 


//$count = $User->where("status=1")->count(): // 查 询 满 足 要 求 的 总 记录 数 
$Page = new Page($count,1); /实例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 
$show = $Page->show(); // 分 页 显示 输出 
// 进 行 分 页 数据 查询 注意 limit 方法 的 参数 要 使 用 Page 类 的 属性 
Slist = $db->order('id'")->limit($Page->firstRow.',". $Page->listRows)->select(); 
$this->assign('select',$list); /赋值 数据 集 
S$this->assign('page',$show); /赋值 分 页 输出 
S$this->display(main); /输出 模板 

} 


定义 verify0 方 法 ， 载 入 ThinkPHP 中 提供 的 验证 码 扩展 类 ， 调 用 buildImageVerify0 方 法 生 
成 验证 码 。 其 代码 如 下 : 


public function verifyO{ 
import("ORG Util. Image"); // 载 入 验证 码 类 
image::buildImageVerify(4,5,'png’,'65",'30','verify'); // 生 成 验证 人 码 

上 


(2) 定位 到 App\Tpl\default 目录 下 ， 创 建 Index 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 
index.html， 载 入 CSS 样式 文件 和 JavaScript 文件 ， 创 建 表 单 ， 完 成 用 户 登录 信息 的 提交 操作 ， 
通过 img 标签 输出 生成 的 验证 码 。 其 关键 代码 如 下 : 

<link href=" ROOT__/Public/Css/style.css" rel="stylesheet" type="text/css" /> 
<js href=" ROOT_ /Public/Js/check.js" /> 
<form name="forml" method="post" action=" URL /index" onSubmit="returmn chkinput(this)" > 
<table width="265" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td class="title" id="td"> 用 户 名 : </td> 
<td><input name="user" type="text" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 密 码 : </td> 
<td><input name="pass" type="password" size="15" /></td> 
</t> 
<tr> 
<td class="title" id="td"> 验 证 码 : </td> 
<td><input type="text" name="validatorCode" size="10" /></td> 
<td><img stre=" APP /Index/verify/" /></td> 
</t> 
</table> 


. 451 。 


(本 > 六 预知 道 的 300 个 癌 是 


</form> 


这 里 应 用 的 验证 码 是 区 分 字母 大 小 写 的 。 


| (3) 在 Index 模块 文件 夹 下 编辑 main html 文件 ， 通 过 模板 引擎 中 的 session 标签 输出 当 
| 前 登录 的 用 户 名 ， 然 后 通过 foreach 标签 循环 输出 模板 变量 传递 的 数据 ， 最 后 输出 模板 变量 传 
| 递 的 分 页 超 链接 。 其 关键 代码 如 下 : 

<table width="405" border="1" cellpadding="1" cellspacing="1" bgcolor="#99CC33" bordercolor= 
| "HFFFFFF"> 

<tr> 

<td colspan="3" bgcolor="#FFFFFF" class="title" align="center"> 当 前 登录 用 户 : {$Think.session 


| admin}</td> 
| </t> 
<foreach name='select item=user > 
<tr class="content"> 
<td bgcolor="#FFFFFF">&nbsp; {$user.id} </td> 
<td bgcolor="#FFFFFF">&nbsp: {$user.user} </td> 
<td bgcolor="#FFFFFF">&nbsp; {$user.address}</td> 
| </tr> 
| </foreach> 
| <tr class="content"> 
<td colspan="3" bgcolor="#FFFFFF">&nbsp; {$page}</td> 
</tr> 
‘</table> 


运行 结果 如 图 15.12 所 示 。 


| 几 户 后 : FS 

| 将 有 :00v0r 

| 答 证 至: VIE aa 

| 时 ro 
| | 一 用户 信息 


1 
5 条 记 好 16 而 下面 


总 淄 则 出 


| 图 15.12 验证 码 类 和 分 页 类 的 应 用 效果 


问题 279 如何 通过 ThinkPHP 中 的 扩展 类 生成 中 文 验证 码 ? 


| 问题 阐述 

| 如何 通过 ThinkPHP 中 的 扩展 类 生成 中 文 验证 码 ? 

| 专家 解答 

| 通过 ThinkPHP 中 的 扩展 类 生成 中 文 验证 码 ， 其 应 用 的 是 验证 码 类 中 的 GBVerify (方法 。 
语法 如 下 ， 


GBVerify ($length,$type,$width,$height, $fontface, $verifyName) 
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GBVerify (方法 的 参数 说 明 如 表 15.3 所 示 。 
表 15.3 GBVerify () 方 法 的 参数 说 明 


的 长 度 ， 默 认为 4 位 数 

的 图 片 类 型 ， 默 认为 png 

的 图 片 类 型 ， 默 认为 png 

的 宽度 ， 默 认 会 自动 根据 验证 码 长 度 自动 计算 
的 高 度 ， 默 认为 50 

的 SESSION 记录 名 称 ， 默 认为 verify 
使 用 的 字体 ， 使 用 完整 文件 名 或 放 到 图 像 类 所 在 的 目录 下 ， 默 认 使 用 的 字体 文件 是 
simhei.tt 


前 | 江 | 到 | 如 | 六 | 二 


在 应 用 GBVerify 0) 方法 生成 中 文 验 证 码 时 ， 必 须 判 断 以 下 几 个 条 件 : 

加 ”PHP 是 否 已 经 安装 GD 库 支持 。 

输出 之 前 是 否 有 任何 的 输出 (尤其 是 UTF-8 的 头 信息 输出 )。 

Jmage 类 库 是 否 正确 导入 。 | 
如 果 是 中 文 验证 码 检查 是 否 有 复制 字体 文件 到 类 库 所 在 目录 。 | 


应 用 示例 
下 面 通过 示例 讲解 如 何 通 过 ThinkPHP 中 扩展 类 生成 中 文 验证 码 。 其 关键 步 又 如 下 : 


在 App\LibWAction\ 目 录 下 , 在 项 目 控制 器 的 Index 模块 中 定义 verify0 方 法 , 载 入 ThinkPHP | 
中 提供 的 验证 码 扩展 类 ， 并 调用 GBVerify 0 方法 生成 中 文 验证 码 。 其 代码 如 下 : | 


public function verifyO { 
import("ORG Util. Image"); // 载 入 验证 码 类 
image::buildImage Verify(4,5,'png','65','30','verify’); // 生 成 中 文 验证 码 
} 
专家 点 评 


要 应 用 ThinkPHP 2.1 版 本 提供 的 Image 扩展 类 完成 中 文 验证 码 的 生成 ， 需 要 对 \Lib\ORG\ | 
Util\mage.class.php 文件 中 386 行 的 “msubstr” 进 行 修改 ， 将 其 修改 为 “String::msubstr”， 咎 | 
则 将 无 法 输出 中 文 验证 码 。 | 


问题 280 ”如 何 实 现 传 递 查询 条 件 的 分 页 ? 


问题 讲述 
如 何 实现 传递 查询 条 件 的 分 页 ? 
专家 解答 


查询 条 件 在 分 页 超 链接 中 的 传递 ， 关 键 是 在 实例 化 Page 类 中 第 三 个 参数 〈$parameter) 的 
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可 
设置 ， 此 参数 定义 的 是 在 分 页 超 链 接 中 传递 的 参数 名 称 和 对 应 的 参数 值 。 其 关键 代码 如 下 : 
| $Page = new Page($count,l,'key=". $key); 
| 第 一 个 参数 是 查询 的 总 记录 数 ; 第 二 个 参数 是 每 页 显示 的 记录 数 ; 第 三 参数 是 在 分 页 超 链 
优 ]” 接 中 传递 的 参数 和 参数 值 。 


本 示例 通过 TinkPHP 框架 开发 一 个 站 内 搜索 的 功能 ,并 且 对 查询 结果 进行 分 页 输出 。 其 具 


| 体 步 又 如 下 ， 
| (1) 创建 005 项 目 根 目录 ， 在 根 目录 下 创建 项 目 文件 夹 App 和 Public 文件 夹 存储 CSS、 
| 图 片 和 JS 脚本 等 文件 。 
(2) 在 005 项 目 根 目录 下 编辑 index.php 入 口 文 件 。 其 关键 代码 如 下 : 

<2php 

define('THINK_PATH', './ThinkPHP"); // 定 义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 

define('APP_NAME', 'App"); // 定 义 项 目 名 称 

define('APP_PATH', '/App’); // 定 义 项 目 路 径 

TIequire(THINK PATH."/ThinkPHP php"); // 加 载 框架 入 口 文 件 

App::run0; /实例 化 一 个 网 站 应 用 实例 

Wo 


(3) 在 正 浏 览 器 中 运行 入 口 文件 ， 自 动 生成 项 目 目 录 。 
(4) 定位 到 App\Conf 目录 下 ,编辑 config.php 文件 ， 通 过 PDO 连接 MySQL 数据 库 。 其 


| 代码 如 下 : 
| <?php 
Tetum array( 
'DB_TYPE'=> 'pdo', 
/注意 DSN 的 配置 针对 不 同 的 数据 库 有 所 区 别 
'DB_DSN'=> 'mysqlhost=localhost:dbname=db_database'， 
DB_USER'=>'root', 
DB _ PWD'=>'111', 
'DB_PREFIX'=>'think ', 
// 其 他 项 目 配 置 参数 …… 
'APP DEBUG' => false, // 关 闭 调试 模式 
和 


(5) 定位 到 \App\Lib\Action\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 Index 模块 ， 继 承 系统 的 
| Action 基础 类 ， 定 义 index() 方 法 。 首 先 ， 获 取 查询 关键 字 〈 通 过 $ POST 获取 表单 提交 的 查询 
| 关键 字 ; 通过 $_GET 获取 超 链接 传递 的 查询 关键 字 ， 这 是 查询 结果 分 页 输出 的 关键 )， 并 且 将 
| 其 赋 给 指定 的 变量 。 其 代码 如 下 : 


<2php 


session_ start(); /初始 化 SESSION 变量 
header("Content-Type:text/html: charset=utf-8"); /设置 页 面 编码 格式 
class IndexAction extends Action{ 
public function indexO{ 
ifisset($_POST[key]){ J/ 浏 断 查询 的 关键 字 是 否 存在 


Skey=$ POST['key']; 
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jelse iftisset($ GET['key']){ 
S$key=$_ GET['key']; 


} | 
然后 ， 实 例 化 基础 模型 类 ， 载 入 分 页 类 。 其 代码 如 下 : | 二 光 
$db = MCUser); /实例 化 模型 类 ,参数 数据 表 名 称 ， 不 包含 前 级 | 天 
import("ORG Util Page"); // 导 入 分 页 类 | 


接着 ， 判 断 查 询 关 键 字 是 否 存在 ， 如 果 存 在 ， 则 根据 获取 的 关键 字 执行 模糊 查询 ， 并 且 对 | 
查询 结果 进行 分 页 输出 ， 如 果 不 存在 ， 则 对 数据 库 中 的 所 有 数据 进行 分 页 输出 。 其 代码 如 下 : 
if($key!="){ 
$map="user like('%".$key."%') or address like('%".$key."%") "; 
S$list = $db->where($map)->order('id desc')->page($_GET['p'].',1")->selectO; 


S$this->assign('select', $list); 1/ 赋值 数 据 集 
$count = $db->where($map)->countO); // 查 询 满足 要 求 的 总 记录 数 
SPage = new Page($count,1,'key=". $key); /实例 化 分 页 类 ， 传 入 总 记录 数 、 每 页 显示 的 
记录 数 和 查询 的 关键 字 
$show = $Page->show(); // 分 页 显示 输出 
S$this->assign('page', $show); // 赋 值 分 页 输出 
}else{ 
$count = $db->count(); // 统 计 总 记录 数 
$Page = new Page($count, 1); // 实 例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 
$show = $Page->show(); // 分 页 显示 输出 


// 进 行 分 页 数据 查询 。 注 意 limit 方法 的 参数 要 使 用 Page 类 的 属性 
$list = $db->order('id')->limit($Page->firstRow.',".$Page->listRows)->select(); 


S$this->assign('select', $list); /赋值 数据 集 
S$this->assign('page',$show): // 赋 值 分 页 输出 

} 

最 后 ， 应 用 display() 方 法 指定 模板 页 。 其 代码 如 下 : 
$this->displayO; /输出 模板 

} 

} 

> 


(6) 定位 到 App\Tpl\default 目录 下 ， 创 建 mdex 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 
index.html， 载 入 CSS 样式 文 ; 创建 表单 ， 提 交 查 询 的 关键 字 ， 应 用 foreach 语句 完成 查询 结果 
的 分 页 输出 ; 应 用 {$page} 输 出 分 页 超 链 接 。 其 关键 代码 如 下 : 


<link href=" _ ROOT_ /Public/Css/style.css" rel="stylesheet" type="text/css" /> 

<form id=-"forml" name="forml" method="post" action=" URL /index"> ! 

站 内 搜索 : 

<input type="text" name="key" id="key" /> | 

<input type="submit" name="button" id="button" value= "提交 " /> 

</form> 

<foreach name='select item='"user’ > ! 
<tr class="content"> 

<td bgcolor="#FFFFFF">&nbsp: {$user.id}</td> | 

<td bgcolor="#FFFFFF">&nbsp; {$user.user} </td> | 

<td bgcolor="#FFFFFF">&nbsp; {$user.address}</td> | 

</t> | 


. 455 。 


人 Pp 六 须知 道 的 300 个 问题 


</foreach> 
<tr class="content"> 
<td colspan="3" bgcolor="#FFFFFF">&nbsp; {$page} </td> 


</tr> 


云 行 结果 如 图 15.13 所 示 。 


| 
用 户 信息 | 

[wm 名 称 地 : | 

I= 长 机 

[条 记录 芭 页 上 二 直下 -页 123 和 4 | 


图 15.13 带 查 询 条 件 的 分 页 


| 站 肉 持 索 : 上 硅 于 
证 


问题 281 ”如何 应用 ThinkPHP 中 的 扩展 类 上 传 文件 ? 


问题 阐述 


如 何 应 用 ThinkPHP 中 的 扩展 类 上 传 文件 ? 


专家 解答 


通过 ThinkPHP 中 的 扩展 类 上 传 文件 应 用 的 是 文件 上 传 扩 展 类 (Lib\ORG\Net\ UploadFile 
-class.php) 中 的 方法 ， 从 而 实现 文件 上 传 功能 。 
在 UploadFile 类 中 可 以 对 上 传 文件 的 属性 〈 参 数 ) 进行 控制 ， 其 可 控 属 性 如 表 15.4 所 示 。 


表 15.4 在 UploadFile 类 中 上 传 文件 的 可 控 属 性 


属 性 描述 
maxSize 控制 上 传 文件 的 大 小 以 字 节 为 单位 )， 默 认为 -1 不 限制 大 小 ) 
控制 上 传 文件 在 服务 器 中 的 存储 位 置 ， 如 果 设 置 为 空 ， 则 使 用 UPLOAD_PATH 常 量 定 
savePath 义 的 路 径 
上 传 文件 的 保存 规则 ， 必 须 是 一 个 无 须 任 何 参数 的 函数 名 , 如 time、uniqid com _create_ 
svenle guid 等 ， 但 必须 保证 生成 的 文件 名 是 唯一 的 ， 默 认 是 uniqid 
hashType 传 文件 的 哈 希 验证 方法 ， 默 认 是 md5_file 
autoCheck 是 否 自动 检测 附件 ， 默 认为 自动 检测 
UploadReplace “| 存在 同名 文件 是 否 是 履 盖 
allowExts 允许 上 传 的 文件 后 级 〈 留 空 为 不 限制 》， 使 用 数组 设置 ， 默 认为 空 数组 
allowTypes 允许 上 传 的 文件 类 型 〈 留 空 为 不 限制 ) ， 使 用 数组 设置 ， 默 认为 空 数组 
thumb 是 否 需 要 对 图 片 文件 进行 缩 略图 处 理 ， 默 认为 false 
thumbMaxWidth | 缩 略 图 的 最 大 宽度 ， 多 个 使 用 逗号 分 隔 
thumbMaxHeight | 缩 略 图 的 最 大 高 度 ， 多 个 使 用 逗号 分 隔 
thumbPrefix 缩 略图 的 文件 前 级 ， 默 认为 thumb_ 
thumbSuffix 缩 略 图 的 文件 后 组， 默认 为 空 
thumbPath 缩 略图 的 保存 路 径 ， 若 留 空 则 取 文件 上 传 目 录 本 身 
thumbFile 指定 缩 略 图 的 文件 名 
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上 传 文件 的 属性 设置 完成 后 ， 应 用 upload() 方 法 完成 文件 上 传 操作 。 | 
文件 上 传 成 功 后 ， 应 用 getUploadFileInfo 方法 获取 附件 信息 。 其 返回 值 是 一 个 数组 ， 数 组 | 
中 元 素 的 含义 如 下 : | 
key: 上 传 的 表单 名 称 。 
savepath: 上 传 文件 的 保存 路 径 。 
name: 上 传 文件 的 原始 名 称 。 
savename: 上 传 文件 的 保存 名 称 。 
size: 上 传 文件 的 大 小 。 
type: 上 传 文件 的 MIME 类 型 。 
extension: 上 传 文件 的 后 绥 类 型 。 
hash: 上 传 文件 的 哈 希 验证 字符 串 。 
果 上 传 失 败 ， 则 应 用 getErorMsg 方法 获取 错误 提示 信息 。 


加 加 网 辐 辐 网罗 加 


全 


注意 : | 
在 创建 文件 上 传 的 表单 时 ,在 Form 标签 中 必须 设置 enctype="multipart/form-data", 否则 文 | 
件 不 能 上 传 。 | 


应 用 示例 | 
本 示例 应 用 ThinkPHP 中 的 文件 上 传 扩展 类 (Lib\ORGWNet\WUploadFile.class.php》 实 现 文件 | 
上 传 的 功能 ， 上 传 jpg、sif 或 png 格式 的 图 片 到 服务 器 指定 的 文件 夹 下 。 其 具体 步 又 如 下 : | 
(1) 创建 006 项 目 根 目 录 ， 在 根 目录 下 创建 项 目 文件 夹 App 和 Public 文件 夹 存储 CSS、 
图 片 和 JS 脚本 等 文件 。 
(2) 在 006 项 目 根 目录 下 编辑 index.php 入 口 文件 。 其 关键 代码 如 下 : 


<2?php 

define(CTHINK_ PATH', '../ThinkPHP'): // 定 义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
define(APP NAME','App); /定义 项 目 名 称 

define('APP_PATH,', '/App)); /定义 项 目 路 径 

Tequire(THINK PATH."/ThinkPHPphp"): /加 载 框架 入 口 文件 

App::run0: /实例 化 一 个 网 站 应 用 实例 

?> 


(3) 在 正 浏 览 器 中 运行 入 口 文 件 ， 自 动 生 成 项 目 目录 。 
(4) 定位 到 App\Conf 目录 下 ， 编 辑 config.php 文件 ， 并 通过 PDO 连接 MySQL 数据 库 。 
其 代码 如 下 : 


<2php 
Tetum array( 
IDB_TYPE'=> 'pdo', 
/注意 DSN 的 配置 针对 不 同 的 数据 库 有 所 区 别 
'DB_DSN'=> ‘mysql:host=localhost:dbname=db_database', 
DB_USER'=>'root, 
'DB_ PWD=>"111', 
'DB_PREFIX'=>'think ', 
// 其 他 项 目 配 置 参 数 …… 
'APP DEBUG' => false, // 关 闭 调试 模式 
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(5) 定位 到 \App\Lib\Action\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 Index 模块 ， 继 承 系统 的 Action 
基础 类 ， 定 义 mdex() 方 法 ， 在 此 方法 中 没有 任何 其 他 操作 ， 直 接 指 定 模板 页 index.html， 在 模 
| 板 页 中 创建 表单 ， 添 加 文件 域 ， 将 图 片 提交 到 upload() 方 法 中 进行 处 理 。index() 方 法 的 代码 如 下 : 
| public function indexO{ 


| S$this->display('index'"); 
ro” 轩 汪 十 


| 在 Index 控制 器 的 Index 模块 中 定义 upload0 方 法 ,实现 文件 上 传 的 功能 。 首 先 , 载 入 上 传 
| 文件 类 ， 实 例 化 上 传 文件 类 ; 然后 ， 对 上 传 文件 的 大 小 、 类 型 、 存 储 路 径 进 行 设置 ， 接 着 ， 调 
| 用 上 传 文件 类 中 的 upload0 方 法 执行 文件 的 上 传 操作 ， 如 果 上 传 成 功 ， 则 将 上 传 文件 的 信息 存 
， 储 到 指定 的 数据 表 中 ， 并 且 在 success.html 模板 文件 中 输出 上 传 图 片 ， 否 则 输出 上 传 失 败 的 错 
| 误 信息 ， 并 且 跳 转 到 index.html 页 。 其 完整 代码 如 下 : 


public function uploadO{ 

import("ORGNet.UploadFile"): // 载 入 上 传 文件 类 

S$upload = new UploadFile(); // 实 例 化 上 传 类 

S$upload->maxSize =3145728; // 设 置 附件 上 传 大 小 

$upload->allowExts = array(jpg', 'gif, 'png', 'jpeg”); // 设 置 附件 上 传 类 型 

$upload->savePath = './Public/Uploads/'; /设置 附件 上 传 目 录 

这 $upload->uploadO) { /上 传 成 功 
S$info= S$upload->getUploadFileInfo(); 
Supfile = M("files"); // 实 例 化 User 对 象 
S$upfile->createO; // 创 建 数据 对 象 
S$upfile->tb_name = $info[0]["savename"]; // 保 存 上 传 图 片 名 称 
S$upfile->tb path = $info[0]["savepath"]; // 保 存 上 传 图 片 路 径 
Supfile->tb_date = date("Y-m-d H:i:s"):; // 保 存 上 传 图 片 时 间 
Supfile->addO; // 写 入 用 户 数据 到 数据 库 
$db = M('files'’); /实例 化 模型 类 ， 参 数 数据 表 名 称 ， 不 包含 前 绷 
$select = $db->order('id desc)->select0O; /执行 查询 语句 
S$this->assign('select', $select): // 模 板 变 量 赋值 
$this->success(" 文 件 上 传 成 功 !"); // 上 传 成 功 

}else{ 
S$info=$upload->getErrorMsg(): /上 传 失败 返回 错误 信息 
S$this->redirect('index',", 5,$info): // 页 面 重 定向 

) 

} 


| (6) 定位 到 App\Tpl\default 目录 下 ， 创 建 Index 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 
| index.html， 载 入 CSS 样式 文 ; 创建 表单 ， 定 义 文件 域 ， 将 上 传 图 片 提交 到 upload0 方 法 中 进行 
| 处 理 。 其 关键 代码 如 下 ; 

| <link href=" ROOT /Public/Css/style.css" rel="stylesheet" type="text/css" /> 

| <fom action=" URL __/upload" method="post" enctype="multipart/form-data" name="forml" id="forml1"> 
<tr> 

| <td colspan="3" bgcolor="#FFFFFF" class="title" align="center"> 

| 选择 文件 : 

| <input name="tb_path" type="file" id="tb_path" size="15" /> 

| <input type="submit" name="button" id="button" value=" 提 交 " /> 
| </td> 

| </tr> 

| </form> 
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件 success.html。 通 过 foreach 标签 循环 输出 模板 变量 传递 的 数据 ， 包 括 上 传 图 片 的 名 称 以 及 上 | 
传 图 片 在 服务 器 中 的 存储 位 置 ;通过 img 标 签 根据 模板 变量 传递 的 名 称 和 路 径 输出 上 传 的 图 片 。 | 


<foreach name="select" item="select" > 
<tr class="title"> 
<td bgcolor="#EFFFFF"> 上 传 图 片 : {$select.tb_ name}</td> 
</tr> 
<tr class="title"> | 
<td bgcolor="#FFFFFF" width="44"><img src=" ROOT {$select.tb path} {$select.tb name}" | 


/></td> 
</tr> 
</foreach> 


运行 结果 如 图 15.14 所 示 。 
| 


选择 文件 : 


上 传 图 片 ，CIMG0917.JPG 


图 15.14 文件 上 传 和 浏览 
问题 282 ThinkPHP 框架 中 的 模板 引擎 


问题 阐述 | 
在 ThinkPHP 框架 中 实现 了 动态 文件 和 静态 页 的 分 离 ， 动 态 文件 自 不 必 说 ， 那 么 在 静态 页 | 
模板 中 是 如 何 完成 数据 的 输出 的 呢 ? 其 应 用 的 又 是 哪 种 模板 引擎 呢 ? | 


专家 解答 


ThinkPHP 内 置 了 一 个 基于 XML 的 性 能 卓越 的 模板 引擎 ThinkTemplate， 这 是 一 个 专门 为 | 
ThinkPHP 服务 的 内 置 模板 引擎 。 具 体 来 说 ，ThinkTemplate 是 一 个 使 用 XML 标签 库 技术 编 的 | 
模板 引擎 ， 支 持 两 种 类 型 的 模板 标签 〈 普 通 标签 和 XML 标签 )， 使 用 了 动态 编译 和 缓存 技术 ， | 
而 且 支 持 自 定 义 标签 库 。 | 

ThinkTemplate 模板 引擎 生成 的 编译 文件 默认 存储 于 Runtime/Cache 目录 下 ,以 模板 文件 的 | 
mq5 编码 作为 缓存 文件 名 保存 。 通 过 ThinkTemplate 模板 引擎 中 的 标签 完成 对 动态 文件 中 数据 | 
的 各 种 操作 ， 其 常用 标签 如 表 15.5 所 示 。 | 

表 15.5 ThinkTemplate 模板 引擎 中 的 常用 标签 
应 用 描述 | 
输出 模板 引擎 中 的 变量 。 注 意 模板 标签 的 { 和 4 之 间 不 能 有 任何 | 
的 空格 ， 和 否则 标签 无 效 | 


标签 名 称 


{$name} 
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标签 名 称 
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续 表 


应 用 描述 


// 输 出 $_SERVER 变 量 
{$Think.server.script_name } 

// 输 出 $_SESSION 变 量 
{$Think.session.session idlmd5 } 

// 输 出 $_GET 变 量 
{$Think.get.pageNumber } 

// 输 出 $8_COOKIE 变 量 
{$Think.cookie.name } 

{$Think.now } // 现 在 时 间 
{$Think.templatelbasename } // 模 板 页 面 


{$ 变 量 |default=" 默 认 值 "} 


<volist name="list" id="vo" offset="5" 
length="10'> 

{$vo.name} 

</volist> 


<volist name="list"” id="vo" offset="5" 
length='10> 

{$vo.name} 

</volist> 


系统 变量 。 除 了 常规 变量 的 输出 外 ， 模 板 引擎 还 支持 系统 变量 
和 系统 常量 以 及 系统 特殊 变量 的 输出 。 它 们 的 输出 不 需要 事先 
赋值 给 某 个 模板 变量 。 系 统 变量 的 输出 必须 以 SThink .打头 ， 并 
且 仍 然 支持 使 用 函数 


默认 值 输出 。 如 果 输 出 的 模板 变量 没有 值 ， 但 是 需要 在 显示 时 
赋予 一 个 默认 值 ， 则 可 以 使 用 default 语 法 。 例 如 : 
{$usernicknameldefault=" 明 日 科技 人 

对 系统 变量 的 输出 也 可 以 支持 默认 值 ， 例 如 : 
{$Think.post.nameldefault=" 名 称 为 空 "} 

Volist 标 签 主要 用 于 在 模板 中 循环 输出 数据 集 或 者 多 维 数组 。 标 
签 参数 如 下 : 

Name: 表示 模板 赋值 的 变量 名 称 ， 因 此 不 可 随意 在 模板 文件 中 
改变 。 

Id: 表示 当前 的 循环 变量 ， 可 以 随意 指定 ， 但 确保 不 要 和 name 
属性 冲突 。 

支持 输出 部 分 数据 ， 如 输出 其 中 的 第 5~15 条 记录 。 

Offset: 表示 记录 的 起 始 位 置 。 

Length: 表示 记录 的 长 度 。 

Mod: 控制 输出 记录 的 奇偶 性 ， 还 可 以 控制 在 指定 的 记录 换行 。 
例如 : 

// 输 出 偶数 记录 

<volist name="list" id="vo" mod="2" > 

<eq name="mod" value="1">{$vo.name}</eq> 

</volist> 

// 控 制 一 定 记 录 的 换行 

<volist name="list" id="vo" mod="5" > 

{$vo.name} 

<eq name="mod" value="4"><br/></eq> 

</volist> 


上 = 


<foreach name="list" item="vo" > 
{$vo.1d} 

{$vo.name} 

</foreach> 


专家 点 评 


foreach 标 签 用 于 循环 输出 ， 它 比 volist 标 签 简洁 ， 但 没有 volist 
标签 那么 多 的 功能 。 另 外 ，foreach 标 签 可 以 对 对 象 进行 遍历 输 
出 ， 而 volist 标 签 通常 用 于 输出 数组 


上 述 只 是 ThinkTemplate 模板 引擎 中 的 一 些 常 用 标签 , 其 完整 内 容 读者 可 以 参考 ThinkPHP 
手册 ， 由 于 篇 幅 所 限 ， 这 里 不 再 獒 述 。 
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问题 283 
问题 284 
问题 285 
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问题 292 
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问题 296 
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如 何 隐 藏 PHP 文件 扩展 名 ? 

如 何在 IIS 服务 器 中 实现 UrlRewrite ( 伪 静 态 ) 功能 ? 
如 何在 Apache 服务 器 中 实现 UrlRewrite ( 伪 静 态 ) 功能 ? 
如 何在 分 页 类 中 实现 UrlRewrite ( 伪 静 态 ) 功能 ? 

在 PHP 中 如 何 动态 生成 静态 页 面 ? 

如 何 应 用 Zend mail 发 送 邮 件 ? 

如 何 解 决 使 用 mail0 函 数 发送 邮 件 时 的 标题 乱码 问题 ? 
如 何 实现 带 附 件 的 邮件 发 送 ? 

如 何 实现 邮件 群发 ? 

自 定 义 在 线 编辑 器 的 制作 

如 何 应 用 CKEditor 网 页 编辑 器 ? 

如 何 利用 WebBrowser 打印 报表 ? 

如 何 调用 Word 自动 打印 指定 格式 的 会 议 记录 ? 

如 何 将 Web 页面 中 的 数据 导出 到 Excel 中 并 自动 打印 ? 
如 何 将 查询 结果 保存 到 Word 中 ? 

如 何 将 查询 结果 保存 到 Excel 中 ? 

如 何 实现 在 线 支付 (工商 银行 ) ? 

如 何 实现 在 线 支付 (支付 宝 支 付 ) ? 


需 
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问题 283 如何 隐 匣 PHP 文件 扩展 名 ? 
全 [| 问题 阐述 


隐藏 PHP 文件 扩展 名 ， 使 其 看 上 去 像 其 他 的 编程 语言 ， 以 此 提高 网 站 的 安全 性 。 下 面 我 
| 们 就 来 探讨 一 下 如 何 实现 。 
| 专家 解答 


要 想 隐藏 PHP 文件 扩展 名 ， 主 要 是 要 对 以 下 文件 进行 修改 : 
(1) 在 php.ini 文件 中 设置 expose php=off， 可 以 减少 获得 的 有 用 信息 。 
(2) 对 Apache 服务 器 中 的 配置 文件 httpd.conf 进行 配置 ， 让 类 似 于 Apache 的 Web 服务 
| 端 调用 PHP 解释 其 他 扩展 名 的 PHP 文件 ， 也 就 是 通过 设置 文件 的 扩展 名 来 误导 攻击 者 。 
| 把 PHP 隐藏 为 另 一 种 语言 
| # 使 PHP 看 上 去 像 其 他 的 编程 语言 
AddType application/x-httpd-php .asp .py .pl 
彻底 隐藏 PHP 


# 使 PHP 看 上 去 像 不 知名 的 文件 类 型 
AddType application/x-httpd-php .bop 133t 
| 专家 点 评 


在 完成 以 上 配置 之 后 ， 要 重新 启动 Apache 服务 器 ， 配 置 才能 够 生效 ， 并 且 要 将 程序 中 的 
| PHP 文件 扩展 名 改 为 以 上 配置 的 扩展 名 。 这 种 通过 隐藏 PHP 来 提高 安全 性 的 手段 虽然 防御 能 


力 不 强 ， 但 操作 起 来 较为 简单 。 另 外 ， 隐 藏 PHP 扩展 名 还 可 以 使 用 生成 伪 静 态 和 真 静 态 页 面 
| 的 方法 来 实现 ， 不 过 实现 的 方法 相对 要 复杂 很 多 。 


问题 284 如 何在 JIS 服务 器 中 实现 
UrlRewrite ( 伤 静 态 ) 功能 ? 
| 问题 阅 述 


PHP 伪 静 态 技术 的 应 用 可 以 提高 PHP 代码 的 安全 性 ， 便 于 用 户 记忆 和 输入 ， 且 易于 被 搜 
| 索引 擎 收录 。 例 如 ，http:/**w/news php?id-11 可 以 变 为 http://***/news-11.html。 这 样 不 但 实现 


开发 标记 隐藏 ,避免 每 个 页 面 都 挂 上 .php 的 后 级 , 而 且 可 以 提高 网 站 的 安全 性 。 那么 在 通过 IIS 
| 服务 器 配置 的 PHP 开发 环境 中 如 何 实现 伪 静 态 的 功能 呢 ? 
| 专家 解答 


IIS 默认 是 不 支持 伪 静 态 功能 的 ， 需 要 单独 安装 插件 。 下 面 介绍 如 何 让 IS 支持 伪 静 态 的 功 
| 能 ， 这 里 以 Windows 2003+IIS6.0 为 例 进行 讲解 ， 具 体 步骤 如 下 : 


第 一 步 : 下 载 IS UrlRewrite 插件 , 到 http://www.helicontech.com/download-isapi_rewrite3.htm,， 
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下 载 一 个 ISAPI Rewrite 3 Lite installation package (精简 版 )。 
第 二 步 : 安装 ISAPI Rewrite3 0048 Lite.msi。 | 
第 三 步 : 配置 IS 后 将 其 打开 ,在 Web 属性 的 “ISAPI 筛选 器 ” 中 添加 位 于 安装 目录 的 | 
HeliconVSAPI Rewrite3 文件 夹 中 的 ISAPI Rewrite.dll， 名 称 可 以 自 定义 。 | 
至 此 ，IIS 配置 完成 ， 下 面 开 发 一 个 PHP 项 目 ， 实 现 伪 静 态 的 功能 。 
运行 位 于 安装 目录 Helicon\ISAPI Rewrite3 文件 夹 中 的 Helicon Managerexe， 编 辑 正 则 表 | 
达 式 。 单 击 “Edit” 按 钮 ， 输 入 正则 表达 式 ， 实 现 对 .php 后 级 的 替换 。 其 定义 的 正则 表达 式 如 下 : 
RewriteRule ^phpinfo html$ phpinfo php 


运行 程序 ， 在 浏览 器 中 输入 http:/127.0.0.1/ [sew wm poo | 
ya GO-Bw 可 
phpinfo.html, PHP 程序 正常 运行 , 如 图 16.1 所 示 。 到 -一 = 


专家 点 评 


在 通过 IS 服务 器 搭建 的 PHP 开发 环境 中 实 
现 伪 静态 的 功能 ， 关 键 是 下 载 ISAPI Rewrite3 
0048_Lite.msi 插件 和 配置 IS 服务 器 ， 还 有 就 是 
正则 表达 式 的 定义 , 即 如 何 实现 后 绥 及 参数 的 替换 。 
通过 正则 表达 式 对 文件 的 后 绥 和 传递 的 参数 进行 匹配 ， 完 成 对 PHP 文件 后 级 的 隐藏 操作 。 
下 面 介绍 了 4 种 常用 的 文件 后 级 及 参数 传递 的 正则 表达 式 编写 方法 ， 供 读者 学 习 和 使 用 。 
RewriteRule ^index\.html$ index.php [L] 
上 述 直 接 将 文件 后 级 php 替换 为 html。 其 蔡 换 时 的 后 绥 可 以 任意 指定 。 
RewriteRule ^famousword-([0-9]+)-([a-z]+)\.html$ famousword.php?page=$1&flag=$2 [L] 
上 述 实 现 对 包含 参数 的 PHP 文件 的 后 级 进行 隐藏 ， 内 容 分 析 : 
其 中 famousword 是 文件 的 名 称 ,“-” 是 参数 直接 的 分 阳 符 ， 通 过 “([0-9]+)” 定 义 第 一 
参数 为 数字 格式 ， 通 过 “([a-z]+)” 定 义 第 二 个 参数 为 字符 串 格式 ， 定 义 文件 后 级 为 .html。 
“$” 符 号 后 面 紧 跟 的 是 PHP 文件 (famousword.php) 以 及 传递 的 参数 (page=$1&flag=$2)。 | 
“[L]” 是 结束 字符 。 它 实现 将 famousword.php 文件 隐藏 为 famousword.html， 并 将 其 参数 | 
page 和 flag 的 值 分 别 使 用 数字 和 字符 串 来 替换 。 


RewriteRule ^famouswordinfo-([0-9]+)\.html$ famouswordinfo.php?wordid=$1 [L] 


上 述 内 容 实 现 将 famouswordinfo.php 文件 隐藏 为 famouswordinfo html, 并 将 参数 wordid 传 | 
递 的 值 使 用 数字 来 蔡 换 。 | 


RewriteRule ^maindowncenter-(.*)\.html$ maindowncenter.php?a=$1 [L] 
上 述 内 容 中 的 “(.*)” 表 示 任 意 字 符 串 ， 也 就 是 说 使 用 任意 字符 串 来 蔡 换 参 数 a 的 值 。 


@ phpiniod 


后 @@ imernet | RE 于 而 ”有 130% 


图 16.1 IIS 下 的 PHP 伪 静 态 应 用 


问题 285 ”如 何在 Apache 服务 器 中 实现 UrlRewrite 
( 伪 静 坊 ) 功能 ? 


问题 阐述 | 
PHP 伪 静 态 是 一 个 非常 不 错 的 技术 ， 通 过 它 的 应 用 可 以 提高 PHP 代码 的 安全 性 ， 且 易于 | 
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被 搜索 引擎 收录 。 那 么 这 项 技术 是 如 何 实现 的 呢 ? 确切 地 说 在 Apache 服务 器 中 是 如 何 实现 
的 呢 ? 
专家 解答 
在 Apache 服务 器 中 实现 伪 静 态 功 能 需要 对 Apache 的 配置 文件 进行 修改 ， 并 且 要 编译 一 
个 .htaccess 文件 ， 在 该 文件 中 通过 正则 表达 式 实 现 对 文件 后 缀 和 传递 参数 的 替换 操作 。 其 具体 
操作 步骤 如 下 : 
(1) 修改 Apache 服务 器 的 配置 文件 httpd.conf。 该 文件 存储 在 Apache 安装 文件 夹 下 的 
conf 文件 夹 中 。 打 开 httpd.conf 文件 ， 定 位 到 如 下 位 置 : 
#[L0adModule rewrite module modules/mod rewrite.so 
将 该 项 前 面 的 “#” 去 掉 ， 启 动 该 项 。 
(2) 查找 httpd.conf 文件 ， 找 到 其 中 的 “AllowOverride” 项 ， 将 其 值 修改 为 All。 
(3) 完成 对 httpd.conf 文件 的 修改 后 保存 该 文件 ， 并 重新 启动 Apache 服务 器 ， 使 修改 生效 。 
(4) 接 下 来 在 项 目 中 创建 一 个 .htaccess 文件 ， 实 现 对 PHP 文件 后 级 的 隐藏 操作 。.htaccess 
文件 的 格式 如 下 : 
RewriteEngine on # 启 动 项 
通过 正则 表达 式 对 文件 的 后 组 和 传递 的 参数 进行 匹配 ， 完 成 对 PHP 文件 后 级 的 隐藏 操作 。 
RewriteRule ^index\.html$ index.php [L] 


应 用 示例 
在 站 内 搜索 中 应 用 伪 静 态 技 术 ， 对 文件 的 主页 和 查询 结果 展示 页 面 使 用 伪 静 态 技 术 。 在 本 
示例 中 通过 面向 对 象 编程 应 用 Smarty+PDO 技术 完成 商品 的 分 页 输出 ,并 实现 站 内 查询 的 功能 ， 
最 终 应 用 伪 静 态 技术 将 文件 的 后 级 替换 为 .html 格式 。 其 编译 的 .htaccess 文件 内 容 如 下 : 
# Helicon ISAPI Rewrite configuration file 
# Version 3.1.0.66 
RewriteEngine On 
RewriteRule ^index.html$ index.php 
RewriteRule ^showcommo.html$ showcommo.php 
在 本 示例 中 ， 对 Smarty 模板 的 配置 、 数 据 库 连 接 、 数 据 库 操作 和 分 页 方法 进行 了 封装 。 
然后 ， 实 例 化 类 文件 ， 调 用 类 中 的 方法 ， 完 成 各 种 操作 。 类 的 实例 化 文件 system.inc.php 存储 
于 示例 的 system 文件 夹 下 ， 其 代码 如 下 : 


<?php 

require("system.smarty.inc.php"); // 载 入 Smarty 模板 配置 类 
require("system.class.inc.php"): // 载 入 数据 库 连 接 、 操 作 和 分 页 类 文件 
$connobj=new ConnDB("mysql","localhost","root","111","db_database",false); /数据 库 连 接 类 实例 化 
$conn=$connobj->GetConnId(); 

$admindb=new AdminDB(); // 数 据 库 操作 类 实例 化 

$seppage=new SepPage(); // 分 页 类 实例 化 

$smarty=new SmartyProject(O): // 实 例 化 Smarty 模板 配置 类 

?> 


其 查询 结果 如 图 16.2 所 示 。 
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EE 忆 全 inernet | 保 关 模式 加 用 ow -| 
图 16.2 Apache 下 的 PHP 伪 静 态 应 用 
关于 示例 的 完整 代码 可 参考 本 书 配套 资源 ， 由 于 篇 幅 所 限 ， 这 里 不 再 袭 述 。 
专家 点 评 
无 论 是 在 IIS 还 是 Apache 服 务 器 中 实现 伪 静 态 功能 ,最 为 关键 的 都 是 如 何 定义 正则 表达 式 ，| 
实现 对 超 链 接 的 替换 。 即 使 是 在 超 链 接 中 传递 了 多 个 参数 ， 也 可 以 正常 地 实现 转换 。 例 如 , 对 | 
下 面 的 这 个 有 两 个 参数 的 超 链接 实现 伪 静 态 功 能 ， 其 超 链 接 如 下 : | 
http://127.0.0.1/problem/index.php?page=1&page_type=&parameter2= 
其 定义 的 正则 表达 式 如 下 : 
RewriteRule^index-([0-91+)-([a-z}+)-([a-z}+)\ htmlS$index php?page=$1&page type=$2&parameter2=$3 [L] | 
正则 表达 式 解析 : 将 文件 的 :php 后 缀 替换 为 .html， 将 参数 page 使 用 数字 进行 替换 ， 将 参 | 
数 page _ type 和 parameter2 使 用 字符 串 进行 替换 。 | 


问题 286 ”如 何在 分 页 类 中 实现 UrlRewrite( 伪 静 态 ) 功 能 ? 


问题 阐述 | 
前 面 我 们 讲解 了 如 何在 IS 和 Apache 服务 器 中 实现 伪 静 态 的 功能 , 那么 是 否 可 以 在 封装 的 

分 页 类 中 应 用 伪 静 态 功 能 呢 ?” 如 果 可 以 ， 那 将 是 一 个 非常 完美 的 分 页 类 。 下 面 就 来 讲解 这 个 非 | 

常 完美 的 分 页 类 的 实现 方法 。 

专家 解答 | 
封装 一 个 支持 伪 静 态 功 能 的 分 页 类 ， 首 先 仍然 要 创建 .htaccess 文件 ， 通 过 正则 表达 式 实现 

对 超 链接 的 转换 操作 。 定 义 的 .htaccess 文件 的 内 容 如 下 : | 
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RewriteEngine On 
| RewriteRule ^index.phtml$ index.php 
| RewriteRule ^index-([0-9]+)-([a-z]+)-([A-Z1+)\.phtml$ index.php?page=$1 [L] 
! RewriteRule ^show_data.phtml$ show_data.php 
RewriteRule ^“show_data-([0-91+)\.phtml$ show_data.php?conn id=$1 [L] 


然后 ， 封 装 分 页 类 ， 其 步骤 如 下 : 
(1) 定义 分 页 类 SepPage， 初 始 化 应 用 的 变量 。 


class SepPage{ 
Var $rs; 
Var $pagesize; 
Var $nowpage: 
Var $array; 
Var $conn; 
Var $sqlstr; 
Var $contentname: 
Var $utits; 
Var $anothersearchstr: 
Var $anothersearchstrs: 
Var $class: 


| (2) 定义 ShowData( 方 法 ， 根 据 参数 传递 的 SQL 语句 〈$sqlstr)、 连 接 标识 〈$conn)、 每 
| 页 显示 的 记录 数 〈$pagesize) 和 当前 页 码 值 (Snowpage)， 生 成 分 页 查询 的 SQL 语句 ， 并 且 通 
| 过 PDO 中 的 预 处 理 语 句 执 行 分 页 查询 ， 然 后 通过 PDO 中 的 fetchAll 语句 获取 查询 结果 集 ， 如 
| 果 查 询 结 果 为 空 则 返回 false， 耕 则 返回 查询 结果 集 数组 。 其 关键 代码 如 下 : 

| function ShowData($sqlstr,$conn,$pagesize,$nowpage){ ”// 定 义 方法 


iftlisset($nowpage) || $nowpage—"") /判断 变量 值 是 否 为 空 
Sthis->nowpage=1; // 定 义 每 页 起 始 页 
else 
S$this->nowpage=$nowpage; 
S$this->pagesize=$pagesize; // 定 义 每 页 输出 的 记录 数 
S$this->conn=$conn; // 连 接 数 据 库 返 回 的 标识 
S$this->sqlstr=$sqlstr: // 执 行 的 查询 语句 


S$offset=($this->nowpage-1)*$this->pagesize; 
$sql=$this->sqlstr." limit $offset, $this->pagesize"; 
| $result=$this->conn->prepare($sql); // 准 备查 询 语句 
| $result->execute(): // 执 行 查询 语句 ， 并 返回 结果 集 
$this->array=$result->fetchAll(PDO::FETCH_ASSOC);// 获 取 结 果 集 中 的 所 有 数据 
if(count($this->array)=—0 || $this->array 一 false) 
Tetum false: 
else 
| Tetum $this->array: 
} 


| (3) 定义 ShowPage0 方 法 ， 根 据 传递 的 参数 定义 分 页 超 链 接 ， 实 现 伪 静态 链接 ， 最 终 返 
| 回 分 页 超 链接 字符 串 。 其 关键 代码 如 下 : 


| function ShowPage($contentname,$utits,$anothersearchstr$anothersearchstrs,$class){ 
| if(!isset($anothersearchstr) || $anothersearchstr 一 "") /判断 变量 值 是 否 为 空 
| S$this->anothersearchstr="a": // 定 义 参数 值 


J *。 466 。 


第 16 章 PHP 项 目 开 发 技 配 一 my 
< 


else 
S$this->anothersearchstr=$anothersearchstr; 


if(!isset($anothersearchstrs) || $anothersearchstrs 一 "") // 判 断 变 量 值 是 否 为 空 
$this->anothersearchstrs="A": /定义 参数 值 

else 
S$this->anothersearchstrs=$anothersearchstrs; 


Sres=$this->conn->prepare($this->sqlstr); // 准 备查 询 语句 

S$res->execute(); /执行 查询 语句 ， 并 返回 结果 集 
S$this->array=$res->fetchAll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 数据 
S$record=count($this->array):; // 统 计 记 录 总 数 
Spagecount=ceil($record/$this->pagesize); // 计 算 共有 几 页 


$str=$contentname."&nbsp:".$record."&nbsp;".$utits."&nbsp:; 每 页 &nbsp:;".$this-> pagesize. 
"&nbsp;".$utits."&nbsp; 第 &nbsp;".$this->nowpage."&nbsp: 页 / 共 &nbsp:;".$pagecount."&nbsp; 页 "; 
$str="&nbsp:&nbsp:&nbsp:&nbsp:"; 
if($this->nowpage!=1) 
Sstr="<a href=" substr($ SERVER[PHP SELF].0.4)""."1"."-" $this->anothersearchstr "~" 
.$this->anothersearchstrs.".phtml"." class=".$class."> 首 页 </a>"; /设置 首页 超 链接 
else 
$str.="<font color=#555555 > 首页 </font>" 
$str.="&nbsp;"; 
if($this->nowpage!=1) 
$str.="<a hre 合 ".substr($_ SERVER[PHP SELF'],0,-4)."-".($this->nowpage-1)."-" 
.$this->anothersearchstr."-".$this->anothersearchstrs.".phtml"." class=".$class."> 上 一 页 </a>"; 
// 设 置 上 一 页 的 超 链接 


else 
$str.="<font color=#555555> 上 一 页 </font>"; 
$str.="&nbsp;"; 
if($this->nowpage!=$pagecount) 
$str.="<a hre 合 ".substr($_ SERVER[PHP SELF'].0.-4)."-".($this->nowpage+l) 
."-".$this->anothersearchstr."-".$this->anothersearchstrs.".phtml"." class=".$class."> 下 一 页 </a>"; 
// 设 置 下 一 页 的 超 链接 


else 
$str.="<font color='#555555 人 > 下 一 页 </font>"; 
$str.="&nbsp;"; 
if($this->nowpage!=$pagecount) 
$str="<a href=".substr($_ SERVER['PHP_SELF'].0.-4)."-".$pagecount."-".$this-> 
anothersearchstr."-".$this->anothersearchstrs.".phtml"." class=".$class."> 尾 页 </a>"; ”// 设 置 尾 页 的 超 链 接 
else 
$str.="<font color=#555555 作 尾 页 </font>"; 
这 count($this->array) 一 0 || $this->array 一 false) 
Tetum "无 数据 ! "; 
else 
Tetum $str; 


} 


至 此 ， 分 页 类 创建 完成 。 此 时 将 其 存储 于 system\system.class.inc.php 文件 中 。 另 外 ， 在 该 
文件 中 还 存储 了 数据 库 的 连接 和 管理 类 以 及 系统 常用 方法 类 ， 由 于 篇 幅 所 限 ， 这 里 不 青 著述 。 
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说 明 : 
| 在 ShowPage() 方 法 中 定义 的 超 链 接 就 是 通过 伪 静 态 生成 的 。 通 过 $ SERVER[PHP SELFI] 
| 获取 当前 脚本 文件 的 名 称 , 然后 应 用 substr() 函 数 对 文件 名 进行 截取 ,返回 没有 后 组 的 文件 名 称 ， 
| 在 配合 方法 传递 的 参数 ， 根 据 伪 静态 中 定义 的 超 链接 格式 重新 定义 超 链接 字符 串 。 


| 然后 在 system 文件 夹 下 创建 system.inc.php 文件 ， 实 现 类 的 实例 化 操作 ， 通 过 实例 化 返回 
的 对 象 调用 类 中 的 方法 ， 实 现 数 据 库 的 连接 、 管 理 以 及 数据 的 分 页 输出 。 
| 运行 结果 如 图 16.3 所 示 。 
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| 图 16.3 在 分 页 类 中 应 用 伪 静 态 功 能 

| 专家 点 评 

| 在 分 页 类 中 实现 伪 静 态 是 伪 静 态 技 术 在 HS 或 Apache 服务 器 下 具体 应 用 的 体现 , 在 应 用 伪 
| 静态 技术 之 前 必须 完成 对 服务 器 的 配置 ， 然 后 才 可 以 进行 具体 的 操作 。 所 谓 在 分 页 类 中 应 用 伪 
| 静态 技术 ， 其 关键 就 是 如 何 定义 通用 的 分 页 超 链接 。 如 何 将 这 个 通用 的 分 页 超 链接 定义 成 伪 静 
| 态 的 格式 ， 其 关键 就 是 5_SERVER['PHP_SELF]、substr0 函 数 和 字符 串 连接 符 的 巧妙 运用 。 


问题 287 在 PHP 中 如 何 动态 生成 静态 页 面 ? 


| 问题 阐述 
| 为 了 有 效 地 提高 站 点 的 访问 效率 、 降 低 服务 器 负载 、 提 高 被 搜索 引擎 搜索 到 的 几率 ， 可 以 
| 在 PHP 中 动态 生成 静态 页 面 的 形式 进行 答 出。 那么 在 PHP 中 如 何 动态 生成 静态 页 面 呢 ? 
专家 解答 
| PHP 动态 生成 静态 页 面 主要 应 用 PHP 中 的 缓存 函数 。 
| GD ob start0 函 数 ， 开 启 缓存 ， 这 时 将 不 会 有 来 自 脚本 的 内 容 被 输出 ， 这 些 内 容 都 存储 
| 在 缓 在 中 。 语 法 如 下 ， 
bool ob start ( [callback output _ callback [, int chunk_ size [, bool erase]]] ) 
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ob_start() 函 数 的 参数 说 明 如 表 16.1 所 示 。 
表 16.1 ob_start() 函 数 的 参数 说 明 
说 明 
可 选 参 数 。 设置 回 叫 函 数 ， 当 遇 到 ob_end_flush0 函 数 时 或 者 在 脚本 末尾 缓存 自动 输出 时 ， 
回 叫 函 数 才 被 执行 。 函 数 会 收 到 以 输出 的 内 容 作为 参数 ， 并 且 函 数 需要 返回 一 个 包括 新 
内 容 的 变量 


参 数 


output_ callback| 


chunk size ”| 可 选 参数 。 设 置 每 行 输出 内 容 的 字 节 数 ， 达 到 设置 的 字 节 数 就 会 换行 | 
erase 可 选 参数 。 如 果 设置 为 blse， 那 么 在 脚本 结束 后 ， 缓 存 不 会 被 删除 ， 作 用 于 php 4.3.0 之 后 | 


(2) ob_end_flush0 函 数 ， 向 浏览 器 输出 缓存 中 的 内 容 ， 并 关闭 缓存 ， 成 功 返 回 tme, 失 | 
败 返 回 false。 | 
bool ob_ end _ flush (void ) 


注意 : | 
如 果 失 败 将 会 出 现 E_ NOTICE 级 别 的 错误 信息 ， 返 回 的 布尔 值 是 php 4.2.0 以 后 添加 的 。 | 
(3) ob_get_contents() 函 数 ， 获 取 缓 存 中 的 内 容 ， 失 败 则 返回 false。 

string ob_get_contents ( void ) | 
(4) 创建 StaticPage 类 ， 应 用 ob_start() 函 数 定义 pageBegin() 动 态 页 面 的 开始 方法 ; 应 | 
用 ob_end flush0 函 数 定义 pageEnd0) 动 态 页 面 的 结束 方法 ， 定 义 getStaticPageName() 方 法 ， | 
根据 当前 请 求 地 址 获取 静态 页 面 的 名 称 ， 定 义 makeStaicPage0 方 法 ， 生 成 静态 页 面 ， 定 义 | 
reMakeCondition() 方 法 ， 在 指定 的 时 间 内 重新 生成 静态 页 面 。 | 
应 用 示例 | 
应 用 PHP 中 的 缓存 函数 ， 创 建 StaticPage 类 ， 实 现 将 动态 文件 生成 静态 页 面 。 有 具体 实现 过 | 
程 如 下 : | 
(1) 本 示例 使 用 Adodb 技术 管理 MySQL 数据 库 ， 为 了 提高 代码 重用 率 、 体 现 PHP 面向 | 

对 象 编程 的 特点 ， 创 建 ConnDB 数据 库 连 接 类 实现 与 MySQL 数据 库 的 连接 ， 代 码 如 下 : 


<2?php 

class ConnDB{ | 
var $dbType; /| 数据 库 类 型 | 
var $host; /MYSQL 数据 库 服务 器 地 址 | 
Var $userName; 1// 用 户 名 | 
Var $passwWord: /密码 | 
Var $dbName: /数据库 名 | 
var $isDebug: // 是 否 调试 | 
var $connID; // 保 存 数据 库 连 接 标识 | 


function ConnDB ($dbType = "mysql', $host, $userName, $password, $dbName, $isDebug = false){ | 
// 构 造 函 数 ， 用 于 对 类 中 的 属性 进行 初始 化 | 
S$this->dbType = $dbType; | 
S$this->host = $host; | 
$this->userName = $userName: | 
S$this->password = $passwWord: | 
S$this->dbName = $dbName: | 
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S$this->isDebug = $isDebug: 


| 有 
| function getConnID 0O{ 
| Tequire_once '\Tibrary/adodb/adodb inc php': // 导 入 Adodb 类 库 
| S$this->connID = NewADOConnection($this->dbType); /建立 连接 
食 半 | | if($this->dbType 一 mysql' | $this->dbType 一 'mssql) {//MySQL 或 SQL Server 数据 库 
| S$this->connID->Connect($this->host, $this->userName, $this->password, $this-> 
es 


下 ($this->dbType — mysql) { 
S$this->connID->Execute('set names gbk'’); // 如 果 是 MySQL 数据 库 ， 则 设置 字符 集 


} 
} elseif ($this->dbType 一 'access') { //Access 数据 库 
| S$this->connID->Connect(Driver = {Microsoft Access Driver (*.mdb)}:; Dbq = . 
| realpath($this->dbName) . :Uid =". $this->userName . ';Pwd =" . $this->password. ':;"); 


} else{ 
return false; 
} 
S$this->connID->debug = $this->isDebug; // 是 否 调试 
Tetum $this->connID; 
} 
function closeConnID |{ /关闭 与 数据 库 的 连接 
@ythis->connID->DisconnectO: 
} 


} 
| 在 ConnDB 类 中 不 仅 可 以 建立 与 MySQL 数据 库 的 连接 ， 而 且 可 以 建立 与 Access 和 SQL 
| Server 数据 库 的 连接 。ConnDB 类 中 首先 使 用 构造 方法 对 类 中 的 属性 进行 实例 化 ， 然 后 创建 
| getConnID( 方 法 用 于 返回 数据 库 连 接 的 ID, 最 后 创建 closeConnID( 方 法 释放 数据 库 连 接 资源 。 
| (2) 创建 基于 Adodb 类 库 的 分 页 类 PageDB， 用 于 实现 数据 的 分 页 显示 。PageDB 类 的 代 


| 人 码 如 下 : 
| <2php 
class PageDB{ 
function pageData ($sql, $connID, $pageSize, $page){ 
$arrayPageInfo = array(): // 定 义 空 数组 
if (isset($page) && $page (=") { // 判 断 页 面 变量 page 的 值 是 否 为 空 
SnowPage =$_GET['page']: // 如 果 不 为 空 ， 则 获取 变量 的 值 

| } else{ 
| SnowPage = 1; // 否 则 为 变量 赋值 为 1 
| » 
| $rs = $connID->PageExecute($sql, $pageSize, SnowPage); // 执 行 分 页 操作 
| S$arrayData = $rs->GetRows(); // 获 取 结 果 
| if (count($arrayData) — 0 || $rs 一 false) { // 如 果 结 果 等 于 0 或 为 false, 则 返回 false 
| return false: 
| } else{ 
| SarrayPageInfo['data'] = $arrayData; // 将 获取 的 结果 定义 到 数组 中 
| SrsAll = $connID->Execute($sql); /执行 SQL 语句 


$countRs = count(SrsAll->GetRowsO): /统计 查 询 结果 

$countPage = ceil($countRs / $pageSize): /计算 总 的 页 数 
$arrayPageInfo['pageSize'] = $pageSize; /将 每 页 显示 的 记录 数 定义 到 数组 中 
S$arrayPageInfo['countRs'] = $countRs:; /将 总 的 记录 数 定义 到 数组 中 
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S$arrayPageInfo['page'] = $nowPage:; // 将 当前 页 码 定义 到 数组 中 
$arrayPageInfo['countPage'] = $countPage; // 将 总 的 页 码 定义 到 数组 中 
S$arrayPageInfo['first] = 1; 

if (Spage> 1) { 


$arrayPageInfo['previous] = $rs->AbsolutePage0 - 1: // 将 上 一 页 的 数据 定义 到 数组 中 


}else{ 
$arrayPageInfo[previous'] = 1; 
if ($page < $countPage) { 
$arrayPageInfo[mext] = $rs->AbsolutePageO + 1; // 将 下 一 页 的 数据 定义 到 数组 中 
}else{ 
$arrayPageInfo[mext] = $countPage; 1/ 剩余 页 码 
} 
S$arrayPageInfo['last'] = $countPage: // 最 后 的 页 码 
return $arrayPageInfo; // 返 回 数组 
} 
} 
> 


PageDB 类 中 定义 的 pageData() 方 法 用 于 返回 一 个 参数 数组 ,该 数组 包含 所 有 分 页 信息 , 例 
如 每 页 显示 的 记录 数 、 总 页 数 、 总 记录 数 及 分 页 导航 链接 具体 页 码 值 等 ， 这 样 只 需 根 据 返回 数 
组 就 可 以 获取 所 有 分 页 信息 ， 为 以 后 进一步 开发 作 好 铺垫 。 

(3) 创建 StaticPage 类 动态 生成 静态 页 面 主要 使 用 技术 要 点 中 介绍 的 PHP 缓存 函数 以 及 
字符 串 函 数 和 文件 系统 函数 。StaticPage 类 的 代码 如 下 : 


class StaticPage{ 

function pageBegin |{ // 指 定 要 生成 静态 页 的 开始 部 分 
ob_start|; 

} 

function pageEnd |O{ // 定 义 要 生成 静态 页 的 结束 部 分 
ob_end_flush(); 

b 
function getStaticPageName ($url, $extendsName, $defaultPage = "index"){ 
// 根 据 动态 地 址 生成 静态 页 的 名 称 
S$baseName = basename($url): // 不 包含 目录 的 php 动态 页 名 


S$pageName = substr($baseName, 0, strpos($baseName, "php")); ” // 去 掉 扩展 名 
还 ($pageName —") { 
SpageName = $defaultPage; 


; 
$searchString = substr(strechr($baseName, '?'"), 1, strlen(strchr($baseName, "?"))); 
// 获 取 查 询 字 符 串 部 分 
$arraySearchString = explode('&', $searchString); /分 割 查询 字符 串 并 保存 到 数组 中 
S$searchName ="; 
for ($i= 0; $i < count($arraySearchString); $i++) {// 使 用 “-” 连 接 查 询 字 符 串 的 值 
$searchStringTmpValue = strstr($arraySearchString[$1], ="); 
if($searchStringTmpValue (=") { 
$searchName = "~ . substr($searchStringTmpValue, 1, strlen($searchStrngTmpValue)); 


!; 
} 
$fileName = $pageName . $searchName .'… . $extendsName; // 生 成 静态 页 文件 名 
return $fileName: // 返 回 静态 页 文件 名 
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上 
function makeStaicPage ($url, SentendsName, $saveDir){ // 生 成 静态 页 文件 
$content = ob_get_contents(); // 获 取 PHP 页 面 生成 的 HTML 页 面 内 容 
下 (is_dir($saveDiD) { 
mkdir($saveDiD): /创建 保存 静态 页 的 目录 
} 
$fileName = $this->getStaticPageName($url $entendsName); 
$ 印 = fopen($saveDir ./ . $fileName, 'w'); 
fwrite($fp, $content); // 将 HTML 内 容 写 入 静态 页 
上 
function reMakeCondition ($saveDirAndFile, SreMakeTime = -1){ 
// 设 定 重新 生成 静态 页 的 条 件 
if (SreMakeTime ==- 1){ 
if (file_exists($saveDirAndFile)) { 
return true; 
} 
} elseif (file_ exists($saveDirAndFile)&& time()-filemtime($saveDirAndFile)<$reMakeTime) { 
// 如 果 静 态 页 超出 指定 的 时 间 范 围 或 不 存在 , 则 返回 tue, 表示 将 重新 生成 静态 页 
return true; 
} 
Teturn false; 
) 


} 

在 StaticPage 类 中 ,定义 pageBegin0 和 pageEnd() 方 法 分 别 用 来 定义 所 要 生成 静态 页 面 的 开 
始 和 结束 位 置 ;定义 getStaticPageName() 方 法 根据 实际 PHP 页 面 的 请 求 地 址 生成 静态 页 面 地 址 ; 
定义 makeStaicPage() 方 法 创建 静态 页 面 ; 定义 reMakeCondition() 方 法 用 于 指定 重新 生成 静态 页 
面 的 条 件 。 

StaticPage 类 中 生成 静态 页 的 方法 是 真正 意义 地 生成 静态 页 面 ， 而 非 伪 静态 ， 其 好 处 是 不 
受 服 务 器 局 限 ， 如 果 使 用 伪 静 态 的 方法 则 要 开启 Apache 服务 器 的 Rewrite 功能 。 另外， 如 果 用 
户 使 用 租用 的 空间 ， 可 能 有 些 空间 提供 商 未 提供 上 述 技术 支持 。 

(4) 创建 index.php 页 面 ， 实 现 数据 的 分 页 输出 。 

首先 ， 在 页 面 最 开始 调用 StaticPage 类 ， 并 根据 reMakeCondition() 方 法 判断 与 当前 PHP 页 
面 所 对 应 的 静态 页 面 是 否 存在 ， 或 者 静态 页 面 是 否 已 经 过 期 ， 如 果 过 期 则 重新 生成 ， 否 则 使 用 
header() 函 数 重 定向 到 与 该 页 面 所 对 应 的 静态 页 面 中 , 并 使 用 exitO 函 数 终止 该 PHP 页 面 剩余 代 
码 的 执行 ， 从 而 提高 代码 执行 效率 。 代 码 如 下 : 


<2php 

require_once 'StaticPage.php'; // 包 含 StaticPage 类 

S$ststicPage = new StaticPage(): // 对 StaticPage 进行 实例 化 
$ststicPage->pageBegin(); /定义 生成 静态 页 面 的 起 始 位 置 
$dirName = '/pages': /定义 生成 静态 页 的 目录 


$fileName = $ststicPage->getStaticPageName($_ SERVER[REQUEST _URIT],'shtml); 
// 生 成 与 PHP 页 面 对 应 的 静态 页 名 称 
if($ststicPage->reMakeCondition($dirName ./. $fileName, 60)){// 判 断 是 否 需 要 重新 生成 静态 页 
header(location:pages/" . $fileName); // 不 需要 则 重新 定向 到 与 该 PHP 页 面 所 对 应 的 静态 页 面 
exitO; 
$baseUrl = VTMV/Imstances/20/17': 
require_once 'ConnDB.php'; // 包 含 数 据 库 连接 类 
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require_once 'PageDB .php': // 包 含 分 页 类 
$connObj = new ConnDB('mysql', 'localhost', 'root', 'root', 'db_database', false); // 实 例 数 据 库 连接 类 
$connID = $connObj->getConnIDO; // 返 回 数据 库 连接 人 DD 


> 


然后 ， 实 例 化 PageDB 分 页 类 ， 通 过 for 循环 输出 数据 ， 并 创建 分 页 超 链接 ， 完成 数据 的 | | 
分 页 显示 。 

最 后 ， 关 闭 数据 库 的 连接 ， 调 用 StsticPage 类 的 makeStaicPage0 方 法 生成 静态 页 ， 本 
StsticPage 类 的 pageEnd() 方 法 指定 生成 静态 页 面 的 结束 位 置 。 


<2php 

$connObj->closeConnIDO; /关闭 与 数据 库 的 连接 
$ststicPage->makeStaicPage($_ SERVER['REQUEST_URT], 'shtml', ',/pages); /生成 静态 页 面 
$ststicPage->pageEnd(); // 指 定 生成 静态 页 面 的 结束 位 置 

?> 


将 PHP 动态 页 面 生成 静态 页 面 ， 其 运行 结果 如 图 16.4 所 示 。 


图 16.4 PHP 动态 生成 静态 页 


问题 288 ”如 何 应 用 Zend mail 发 送 邮件 ? 


问题 阐述 | 

目前 很 多 的 网 站 都 支持 邮件 的 发 送 功能 ， 如 注册 用 户 激活 、 用 户 密码 找 回 功能 等 都 可 以 通 
过 邮箱 来 实现 。 那 么 这 个 邮件 的 发 送 功能 是 如 何 实现 的 呢 ? 这 就 是 我 们 要 解决 的 问题 ， 通 过 | 
Zend Framework 框架 中 的 zend_mail 组 件 实现 邮件 的 发 送 功能 。 | 
专家 解答 


既然 是 通过 ZendFramework 框架 中 的 组 件 完成 邮件 的 发 送 功能 ， 那 么 首先 就 要 下 载 | 
ZendFramework 框架 (版 本 为 ZendFramework-1.11.9)。 | 
然后 ， 将 library 文件 夹 下 的 Zend 目录 (组件 文 件 夹 ) 复制 到 项 目 指定 目录 下 。 | 
最 后 ， 在 项 目 文件 中 载 入 发 送 邮件 所 需 组 件 (Zend/Mailphp、Zend/Mail/Transport/ | 
Smtp php)， 调 用 其 中 的 方法 完成 邮件 的 发 送 操作 。 其 具体 方法 如 下 : | 
(1) 通过 include 语句 载 入 所 需 组 件 。 
(2) 获取 表单 提交 的 邮件 信息 ， 包 括 发 件 、 收 件 地 址 以 及 邮件 标题 、 内 容 和 附件 等 信息 。 
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(3) 实例 化 邮件 发 送 类 ， 并 且 设 置 其 编码 格式 为 UTF-8。 
$mail= new Zend MailCUTF-8); /实例 化 发 送 邮件 对 象 
(4) 定义 登录 邮件 服务 器 所 需 的 配置 信息 数组 ， 内 容 包 括 邮箱 名 称 、 密 码 、SSL 和 端口 号 。 
| $config = array(auth' => ‘login','username' => ‘mrsoft8888','password' => ‘mrsoft8888','ssl’ => 
,i'ssl," =>'465"); /定义 SMTP 的 验证 参数 
(5) 实例 化 邮件 登录 验证 类 ， 其 参数 为 登录 邮件 服务 器 的 smtp 地 址 以 及 登录 邮箱 的 数据 ， 
| 即 上 面 定义 的 数组 内 容 。 
| $transport = new Zend Mail Transport Smtp(smtp.sohu.com', $config); /实例 化 验证 的 对 象 


| (6) 判断 表单 提交 的 邮件 信息 中 是 否 包 含 附件 ， 如 果 包 含 附 件 ， 则 通过 下 面 的 方法 获取 
| 表单 提交 的 附件 信息 ， 并 且 调用 setMimeboundary0 方 法 定义 附件 的 内 容 。 其 关键 代码 如 下 : 
| iftisset($_FILES['accessories']['name']) { /判断 是 否 包含 附件 

$fileName = $_FILES['accessories']['name']; 

$file = $_FILES['accessories']['tmp_name']: 

S$fileType = $_FILES['accessories']['type']: 

$contents = file_get contents($file); //PHP 必须 大 于 4.3 才能 使 用 
| // 指 定 mime 类 型 和 处 理 方式 DISPOSITION_ATTACHMENT 附件 形式 、 编 码 风格 、 文 件 名 
| 称 。 其 中 ，feName 的 默认 值 是 null， 我 们 必须 给 出 
| $mail->createAttachment($contents, $fileType, Zend Mime::DISPOSITION ATTACHMENT, 
| Zend Mime::ENCODING BASE64, $fileName); 
| $mail->setMimeboundary(md5(time() + "any")); 

) 
| 通过 $_FILES[] 全 局 数组 获取 附件 的 数据 ; 通过 fie_get_contents() 将 附件 中 的 数据 读 入 到 一 
| 个 字符 串 中 。 
| 通过 createAttachment0 方 法 将 上 传 的 附件 附加 到 邮件 中 。Zend_Mail 会 默认 认为 该 文件 是 
-“ 进 制 对 象 (application/octet-stream)， 以 base64 编码 传输 ， 并 且 作 为 邮件 的 附件 处 理 。 

| 通过 setMimeBoundary() 方 法 设置 特定 的 MIME 分 界线 。 因 为 在 一 个 包含 多 个 段 的 邮件 中 
| 用 于 分 隔 邮 件 不 同 段 的 MIME 分 界线 (MIME boundary) 通 常 是 随机 生成 的 ， 如 果 想 要 通过 特定 
| 的 MIME 分 界线 进行 控制 ， 那 么 就 需要 用 到 此 方法 。 
(7) 定义 邮件 的 主题 、 内 容 和 发 件 人 和 收 件 人 。 


| $mail->setBodyHtml($mailbody): /发 送 邮件 主体 

| $mail->setFrom($form, 'Zend_ Mail 邮件 发 送 测试 7); /定义 邮件 发 送 邮 箱 

| $mail->addTo($to, 'Zend_Mail 邮件 发 送 测试 9: /定义 邮件 的 接收 邮箱 
S$mail->setSubject($subject); // 定 义 邮件 主题 


应 用 Zend_mail 组 件 中 的 基本 属性 ， 实 现 邮 件 信息 的 发 送 ， 其 应 用 属性 如 表 16.2 所 示 。 
| 表 16.2 Zend_Mail 发 送 邮件 的 基本 属性 


| 值 
| addTo($option,$to) 


收 件 人 地 址 ，$option 为 发 件 人 地 址 ，$to 为 邮件 中 发 件 人 地 址 显示 
发 件 人 地 址 ，$option 为 收 件 人 地 址 ，$from 为 邮件 中 收 件 人 地 址 显示 
邮件 标题 

邮件 内 容 ， 文 本 格式 的 数据 

邮件 内 容 ，HTML 格 式 的 数据 
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(8) 调用 send0 方 法 执行 邮件 的 发 送 。 
$mail->send($transport); 
?> 


应 用 示例 | 
创建 一 个 搜狐 邮箱 (mrsoft8888@sohu.com， 密 码 为 mrsoft8888)， 登 录 搜 狐 邮 箱 服务 器 (smtp ， 
.Sohu.com)， 设 置 其 端口 号 为 465, 通过 zend_mail 组 件 实现 邮件 的 发 送 功 能 。 其 关键 代码 如 下 : 


<?php | 
include ‘Zend/Mail php'; // 调 用 发 送 邮 件 的 文件 | 
include 'Zend/Mail/Transport/Smtp.php'; /调用 SMTP 验证 文件 | 
$form="mrsoft8888@sohu.com"; // 定 义 登 录 使 用 的 邮箱 | 
$to=$_POST['to']; // 获 取 邮 件 接收 地 址 | 
$subject=$_ POST['subject]; /获取 邮件 标题 | 
Smailbody=$_POST['content']; // 获 取 邮 件 内 容 | 
$mail = new Zend_ MailCUTF-8)); /实例 化 发 送 邮件 对 象 | 


$config = array(auth' => ‘login','username' => ‘mrsoft8888','password' => ‘mrsoft8888','ssl' => 'ssl', 
port => '465"); /定义 SMTP 的 验证 参数 | 
$transport = new Zend_ Mail Transport Smtp(smtp.sohu.com', $config);。 /实例 化 验证 的 对 象 
iflisset($_ FILES['accessories']['name']) { /判断 是 否 包含 附件 
$fileName = $_FILES['accessories']['name']: 
S$file= $_FILES['accessories']['tmp_name']: 
$fileType = $_FILES['accessories']['type']; 
$contents = file_ get contents($file); //PHP 必须 大 于 4.3 才能 使 用 | 
/指定 MIME 类 型 和 处 理 方式 DISPOSITION_ATTACHMENT 附件 形式 、 编 码 风格 、 文 件 | 
名 称 。 其 中 ， 和 名 eName 的 默认 值 是 null， 我 们 必须 给 出 | 
$mail->createAttachment($contents，$fileType，Zend Mime::DISPOSITION ATTACHMENT, | 
Zend Mime::ENCODING _ BASE64, $fileName); | 
$mail->setMimeboundary(md5(timeQ + "any")); | 
} 


$mail->setBodyHtml($mailbody): // 发 送 邮 件 主体 | 
$mail->setFrom($form, 'Zend_Mail 邮件 发 送 测试 ?; /定义 邮件 发 送 邮箱 | 
$mail->addTo($to, 'Zend_Mail 邮件 发 送 测试 "); // 定 义 邮 件 的 接收 邮箱 | 
$mail->setSubject(Ssubject): /定义 邮件 主题 | 
S$mail->send($transport): | 
?> ! 


运行 结果 如 图 16.5 所 示 。 


图 16.5 邮件 发 送 
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| 关于 示例 的 完整 代码 可 参考 本 书 配 套 资源 ， 由 于 篇 幅 所 限 ， 这 里 不 再 著述 。 
| 专家 点 评 
4 在 通过 Zend _ mail 组件 发 送 邮件 时 ， 除 了 项 目 本 身 所 应 用 的 各 种 技术 方法 外 ， 还 有 一 点 必 
对 J | 须 注意 ， 因 为 它 登 录 的 是 一 个 指定 的 邮箱 ， 并 且 通 过 SMTP 对 登录 的 邮箱 进行 验证 ， 只 有 登录 
成 功 后 才能 通过 该 邮箱 向 其 他 邮箱 发 送 邮件 ， 所 以 我 们 必须 先 注册 一 个 邮箱 ， 然 后 确定 邮箱 对 
应 的 SMTP 服务 器 ， 最 终 确 定 邮 件 登 录 配 置 数据 ， 应 用 zend_mail 组 件 中 方法 实现 邮件 发 送 


问题 289 ”如 何 解决 使 用 mail(0) 函 数 发 送 邮 件 时 
的 标题 乱码 问题 ? 


| 问题 阐述 

| PHP 程序 使 用 mail0 函 数 发 送 邮 件 时 ， 在 接收 到 的 邮件 的 标题 中 的 中 文字 符 串 经 常会 出 现 

| 乱码 。 如 何 解决 这 个 问题 呢 ? 

专家 解答 

在 接收 到 的 邮件 标题 中 ， 中 文字 符 串 出 现 乱 码 ， 追 根 溯源 是 在 发 送 邮 件 时 没有 应 用 

，base64_encode() 函 数 对 字符 串 进行 编码 导致 的 。 

| base64_encode() 函 数 ， 应 用 base64 对 data 进行 编码 。 编 码 后 的 数据 要 比 原始 数据 多 占用 

| 1/3 左右 的 空间 。 函 数 语法 如 下 : 

| string base64_ encode( string data ) 

| 应 用 base64 对 data 进行 编码 。 应 用 base64 编码 后 的 字符 串 只 包含 英文 字母 大 小 写 、 阿 拉 

| 伯 数 字 、 加 号 与 反 斜 线 ， 共 64 个 基本 字符 ， 不 包含 其 他 的 特殊 字符 ， 因 此 ， 被 命名 为 base64。 
通过 mailO 函 数 发 送 邮 件 的 正确 方法 如 下 : 


$mailfom=$_ POST[mailfrom']; /获取 发 件 人 地 址 
$mailto=$_ POST[mailto']; // 获 取 收 件 人 地 址 
$mailcc=$_ POST[mailcc']: /获取 抄 送 人 地 址 
$mailbcc=$_ POST[mailbcc']; /获取 密 送 人 地 址 
$mailsubject=base64 encode($ POST['mailsubject']); 1/ 获取 邮件 主题 
Smailbody = base64_encode($_ POST[mailbody]): 1/ 获取 邮件 内 容 


Sheaders = "From:$mailfrom \r\n Cc:$mailcc \n Bcc:$mailbcc": /定义 邮件 头 

S$headers .= "Reply-To:$mailfrom\\n"; 

$headers = "Content-type: text/html:\r\n charset=iso-8859-1 mn" 

壕 @mail($mailto, $mailsubject,$mailbody.$headers)){// 通 过 mail0 函 数 发 送 邮 件 ， 并 给 出 邮件 发 送 结果 
echo "<script>alert( 邮 件 发 送 成 功 ! ");history.backQ</script>"; 

}else{ 
echo "<script>alert( 邮 件 发 送 失 败 ! ");history.backQ</script>"; 

} 


| 在 应 用 了 base64_encode(O) 函 数 对 标题 和 内 容 进行 编码 后 , 输出 的 中 文字 符 串 就 不 会 出 现 乱 
| 码 了 。 既 然 对 输出 的 字符 串 进行 了 编码 ， 那 么 在 输出 时 就 需要 应 用 对 应 的 函数 对 其 进行 解码 ， 
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这 样 才能 保证 数据 的 正常 输出 ， 其 解码 应 用 的 是 base64_decode0) 函 数 。 | 
base64_decode0 函 数 对 使 用 MIME base64 编码 的 数据 进行 解码 ， 并 返回 原始 数据 ， 返 回 的 | 
数据 可 能 是 中 文字 符 串 或 二 进 制 数 据 ， 失 败 则 返回 false。 函 数 语法 如 下 : | 
string base64 decode ( string encoded data ) 


问题 290 ”如 何 实 现 带 附件 的 邮件 发 送 ? 


问题 阐述 | 
在 实现 邮件 发 送 功能 的 同时 ， 也 可 以 实现 附件 的 发 送 功能 。 通过 附件 可 以 实现 更 多 有 价值 | 
信息 的 传递 。 那 么 这 个 附件 的 发 送 功能 是 如 何 实现 的 呢 ? 下 面 我 们 就 来 探讨 一 下 这 个 问题 。 


专家 解答 


发 送 带 附件 的 邮件 并 不 是 简单 地 将 附件 内 容 上 传 到 POP3 服务 器 的 某 个 目录 下 ， 而 是 首先 | 
将 附件 进行 二 进 制 编码 ， 之 后 将 该 编码 连接 到 邮件 内 容 之 后 ， 当 然 邮 件 内 容 和 附件 编码 之 间 需 | 
要 用 标记 进行 分 割 ， 该 标记 在 邮件 头 中 定义 。 下 面 内 容 为 带 附 件 的 邮件 组 织 形式 : | 

Received: from mrkj ([192.168.1.227]) by MRZTS with Microsoft SMTPSVC(6.0.3790.0); 

Mon, 18 Dec 2006 09:02:40 +0800 

Date: Mon, 18 Dec 2006 00:57:26 +0000 

Subject: 12 

To: zts***(@163.com 

Content-type: multipart/mixed; Boundary= 4585e7760018e 

From:zts***(@163.com 

Return-Path: zts***(@163.com 

Message-ID: 

X-OriginalArrivalTime: 18 Dec 2006 01:02:40.0187 (UTC) FILETIME=[372680B0:01C72240] 

--4585e7760018 

Content-type: text/plain; charset=iso-8859-1 

Content-transfer-encoding: 8bit 

(此 处 为 邮件 内 容 ) 

--4585e7760018 

Content-type: text/plain; name=shop.txt 

Content-disposition: attachment:; filename=shop.txt 

Content-transfer-encoding: base64 

wffV4rXEtcS1xLXEtcS1xLTvtb21xNK7ICC13LXctcS1xA 一 (此 处 为 附件 的 base64 编码 ) 

--4585e7760018-- 


带 附件 的 电子 邮件 的 不 同 部 分 之 间 用 分 界线 来 分 隔 ， 分 界线 在 Content--type 头 中 定义 ， 例 | 
如 “Boundary= 4585e7760018e”， 其 中 “4585e7760018e” 表 示 不 同 部 分 的 分 界线 。 邮 件 的 每 个 | 
新 部 分 以 两 个 连 字号 (--) 和 分 界线 开始 ， 最 后 一 个 分 界线 后 也 有 两 个 连 字 号 ， 表 示 这 个 邮件 | 
中 没有 其 他 部 分 了 。 在 每 个 分 界线 后 有 一 些 行 ， 用 来 告诉 邮件 程序 这 部 分 内 容 的 类 型 ， 如 上 面 | 
例子 中 第 一 个 分 界线 后 面 的 以 Content-type: text/plain 开头 的 行 ， 这 些 行 说 明 后 面 的 部 分 是 ， 
ISO-8859-1 字符 集 的 纯 文 本 形式 , 跟 在 第 二 个 分 界线 后 的 行 告诉 邮件 程序 现在 的 部 分 是 一 个 txt | 
文件 ， 它 的 名 字 是 “shop.txt”。Content-disposition 行 告 诉 邮 件 程序 如 果 可 能 就 以 内 嵌 的 方式 显 | 
示 附 件 ， 现 在 新 的 邮件 程序 会 在 消息 后 显示 txt 文件 的 内 容 ， 如 果 Content-disposition 被 设 为 | 
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attachment， 那 么 邮件 程序 就 不 会 显示 txt 文件 的 内 容 ， 而 是 显示 一 个 连接 到 文件 的 图 标 ， 收 件 
人 要 看 附件 的 内 容 ， 必 须 单 击 这 个 图 标 。 一 般 情况 下 ， 如 果 附 件 是 一 些 文本 ， 则 Content- 

| disposition 就 会 被 设 为 mline, 这 是 因为 现在 大 部 分 邮件 程序 能 够 不 借助 其 他 浏览 器 而 直接 显示 
会 内 | 附件 (文本 ) 的 内 容 。 如 果 附件 不 是 文本 如 图 片 或 其 他 类 似 的 内 容 )，Content-disposition 就 
和 | 设 为 attachment。 


应 用 示例 
应 用 mail0 函 数 实现 邮件 的 发 送 功能 ， 并 且 支 持 附件 的 发 送 。 其 关键 代码 如 下 : 


元 | 二 本 本 本 林 本 本 二 本 二 本 二 本 机 本 本 本 本 本 本 本 二 二 本 本 本 本 志 本 本 本 SET 站 站 本 本 本 本 二 本 二 本 本 本 本 本 本本 本 本 本 本 本 本 本 本 本 本 本 本 本 机 本 本 本 本 本 机 二 > 


<?php 
| $mailffom=$ POST[mailfrom']: /获取 发 件 人 地 址 
| $mailto=$_POST['mailto']; // 获 取 收 件 人 地 址 
$mailcc=$_ POST[mailcc']: /获取 抄 送 人 地 址 
$mailbcc=$_ POST[mailbcc']; // 获 取 密 送 人 地 址 
$mailsubject=base64_encode($_ POST[mailsubject]); // 获 取 邮 件 主题 
$mailbody = base64_encode($_ POST[mailbody]); /获取 邮件 内 容 


这 empty($_ FILES[upload file'][name'])){ 
$headers = "From:$mailfrom \r\n Ce:$mailcc \r\n Bcc:$mailbcc"; /定义 邮件 头 
$headers .= "Reply-To:$mailffomm'": 
$headers .= "Content-type: text/html;\r\n charset=iso-8859-1 \in"; 


if(@mail($mailto, $mailsubject, $mailbody,$headers)) { // 通 过 mailO 函 数 发 送 邮 件 ， 
并 给 出 邮件 发 送 结果 
echo "<script>alert(' 邮 件 发 送 成 功 ! ");history.backQ</script>"; 
jelse{ 
echo "<script>alert(' 邮 件 发 送 失败 ! ");history.backQ</script>"; 
} 
}else{ 
S$file = $_FILES['upload file]['tmp_name']; 1/ 获取 附件 名 称 
$boundary = uniqid("™"); /定义 分 界线 


$headers = "MIME-Version: 1.0 \n"; 
$headers .= "Content-type: multipart/mixed; Boundary= $boundary \r\n"; 


S$headers = "From:$mailfrom\\n"; // 定 义 邮 件 头 
if($_FILES["upload file][type]){ 1/ 判断 附件 类 型 
$mimeType = $_FILES['upload file]['type]:; 
Jelse{ 
$mimeType ="application/unknown": /获取 附件 类 型 
} 
| $fileName = $_FILES['upload file][mname']; // 获 得 附件 名 称 
| $fp = fopen($file, "1"): /打开 文件 
| Sread = fread($fp, filesize($file)): /将 附件 读 入 变量 $read 
| Sread = base64_encode($read): // 转 换 base64 编码 
| Sread = chunk split($read); // 将 长 字符 串 切 成 由 每 行 76 个 字符 组 成 的 小 块 
上 定义 邮件 体 */ 


$body = "--$boundary—— 
Content-type: text/plain; charset=iso-8859-1 
$mailbody 
--$boundary-- 
Content-type: SmimeType: name=$fileName 
Content-disposition: attachment; filename=$fileName 
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Content-transfer-encoding: base64 
Sread 
-$boundary-—"; 
if(mail($mailto, $mailsubject, $body,$headers)){ /用 mail0 函 数 发 送 邮件 
echo "<script>alert(' 附 件 发 送 成 功 ! ):historybackO</script>": 
jelse{ 
echo "<script>alert(' 附 件 发 送 失败 ! ):historybackO</script>": 
} 
} 


?> 
专家 点 评 
当 所 发 送 的 邮件 中 带 有 附件 时 ， 首 先 需要 将 邮件 的 MIME 的 Content-type 类 型 定义 成 
multiparVmixed， 然 后 定义 邮件 内 容 分 割 线 (一 串 字符 或 数字 )， 最 后 将 文件 内 容 编写 在 邮件 主 | 
体内 容 中 进行 发 送 。 | 
注意 ，PHP 的 mail0 函 数 只 能 将 邮件 发 送 到 SMTP 的 邮件 转发 器 中 ， 然 后 通过 转发 器 将 邮 | 
件 发 送出 去 。 | 


问题 291 如 何 实现 邮件 群发 ? 


问题 前 述 | 

人 们 在 工作 中 经 常会 将 相同 内 容 邮 件 发 送 给 多 个 用 户 ， 如 果 一 一 发 送 ， 将 会 给 工作 带 来 很 | 
大 的 麻烦 ， 也 许 还 会 带 来 经 济 上 的 损失 ， 为 了 使 工作 更 加 快捷 、 方 便 和 降低 出 错 率 ， 可 使 用 邮 | 
件 群发 功能 。 那 么 如 何 才 能 实现 邮件 群发 呢 ? | 


专家 解答 | 
要 实现 邮件 群发 主要 应 用 mail0) 函 数 ， 原 理 是 通过 循环 将 邮件 一 一 发 送 ， 关 键 是 如 何 解读 
以 “|” 分 隔 的 多 个 “ 收 件 人 ”邮箱 。 其 主要 应 用 函数 explode()。explode() 函 数 的 语法 如 下 : 
array explode ( string separator, string string [, int limit]) 
explode() 的 参数 说 明 如 表 16.3 所 示 。 


表 16.3 explode() 的 参数 说 明 


参数 


separator 


说 明 : | 
若 参 数 limit 被 设置 ， 则 返回 数组 中 最 多 包含 limit 个 元 素 ， 而 最 后 那个 元 素 将 包含 参数 
String 的 剩余 部 分 ; 如 果 limit 为 负数 , 则 返回 除了 最 后 -limit 个 元 素 外 的 所 有 元 素 ,已 是 PHP 5.1.0 | 


中 新 增 的 特性 ， 用 户 使 用 时 应 注意 PHP 版 本 。 
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| 参数 limit 是 PHP 4.0.1 中 新 增 的 。 由 于 历史 原因 ， 虽 然 implode() 函 数 可 以 接收 两 种 参数 
| 顺序 ， 但 是 explode() 不 行 ， 必 须 参 数 Separator 在 参数 String 之 前 。 


贸 |] |。 应 用 示例 


下 面 通过 一 个 具体 的 示例 来 讲解 邮件 群发 的 实现 方法 。 其 具体 实现 过 程 如 下 : 
(1) 创建 index.php 文件 ， 完 成 邮件 群发 页 面 的 设计 。 
| (2) 在 index.php 页 面 中 根据 提交 按钮 的 值 判断 用 户 是 否 提交 表单 。 通 过 explode() 函 数 
| 将 获取 的 收 件 人 以 “|” 分 割 为 N 个 元 素 ， 以 for 循环 将 得 到 的 N 个 元 素 一 一 通过 mail0 函 数 发 
| 送 。 其 关键 代码 如 下 : 
| if($_POST[Submit]i=""){ 
/开始 得 到 mail 函数 中 的 $to 分 隔 为 N 个 
S$tomany=explode("|",$to); 
/然后 做 一 个 简单 的 判断 : 如 果 只 发 送 了 一 个 邮件 那么 直接 发 送 
for($i=0;$i<count($mailnum);$i++){ 
if(@mail($tomany[$1],$title,$content, $headers)){ 
echo"<script>alert(' 邮 件 发 送 成 功 ! ");</script>"; 
}else{ 
echo"<script>alert(' 邮 件 发 送 失 败 ! ");</script>"; 
b 
| 


邮件 群发 的 运行 结果 如 图 16.6 所 示 ， 将 多 个 邮箱 间 以 “|” 分 隔 ， 同 时 将 同一 邮件 发 送 给 


| 多 个 用 户 。 
| 


收 件 人 : [crm31008163. con 
群 上 邮 昨 中 可 以 格 “ 收 件 人 ”名 称 之 间 以 “|"” 分 后 
发 作 人 : cym31008163. con|264431008@163. com 


图 16.6 邮件 群发 
问题 292 ” 自 定 义 在 线 编 辑 器 的 制作 


| 问题 阐述 


| 在 线 编辑 器 的 功能 经 常会 出 现在 论坛 、 博 客 或 微 博 中 ， 其 作用 是 让 用 户 发 布 的 内 容 更 加 丰 
| 富 多 彩 。 但 是 ， 不 知道 大 家 是 否 注意 到 ， 在 运行 一 些 带 有 在 线 编辑 器 功能 的 页 面 时 ， 在 线 编辑 
| 器 功能 的 加 载 有 时 会 很 慢 ， 网 速 固然 是 一 个 重要 的 原因 ， 但 是 由 于 在 线 编辑 器 功能 的 强大 ， 导 
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致 加 载 缓慢 ， 这 个 原因 也 不 能 忽视 。 那 么 ， 我 们 是 否 可 以 模拟 目前 比较 流行 的 FCKediter 在 线 | 
编辑 嚣 , 编写 一 个 属于 自己 的 在 线 编辑 器 呢 ? 这样 我 们 就 可 以 根据 需求 对 这 个 编辑 器 进行 量 身 | 
定做 ， 既 可 以 满足 我 们 的 需求 ， 又 能 够 对 其 进行 精简 ， 让 它 在 加 载 过 程 中 更 加 清晰 、 流 畅 。 


专家 解答 


自 定义 在 线 编辑 器 其 实 就 是 目前 已 经 成 熟 的 在 线 编 辑 器 的 精简 版 , 其 应 用 jQuery 技术 , 将 | 
操作 方法 封装 到 自 定义 函数 中 ， 最 终 存 储 于 LzhEditorjs 文件 中 。 其 封装 的 部 分 方法 如 下 : 
/可 可 
* 执行 命令 | 
六 1 
function lzhEditorFormat(he, pa) { | 
window.frames["editor"].focus(); 
window.frames["editor"].document.selection.createRange(); 
if (pa—") { 
window.frames["editor"].document.execCommand(hc, false):; 
} else { 
window.frames["editor"].document.execCommand(hc, false, pa); 
} 


} 
ess 
* 创建 链接 
wl 
function lzhEditorCreateLink() { 
var linkStr = prompt("\u8bf7\u8f93\u5165\u94fe\n63a5URL\u5730\5740\uff01", 
if (linkStr != nul) { 
lzhEditorFormat("CreateLink", linkStr); 
} 


7 
* 显示 添加 表情 层 
ead 
function lzhEditorAddFaceO { 
$("#faceLayer").css("display", "block"): 
* 插入 表情 | 
ad 
function insertFace(faceUrD { 
if (faceUr] != null && faceUrl !(="") { | 
lzhEditorFormat("InsertImage", faceUrD); | 


} 
$("#faceLayer").css("display", "none"); | 
} 
/ne* | 
* URL 方式 添加 图 片 
oh 
function lzhEditorInsertImage() { 
if($.trim($("#imageUrl").val0)— "") { 
alert(" 请 输入 要 插入 图 片 的 URL 地 址 ! "); 
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} else{ 

Var V= $.trim($("#imageUrl").valO); 

Var extsName = v.substring(v.lastIndexOf(".") + 1).toLowerCase(); 

if(!(extsName 一 "gif || extsName — "jpg" || extsName — "png" || extsName — "bmp")) { 
alert(" 插 入 的 图 片 只 能 为 gf、jpg、png 或 bmp 格式 ! "); 

}else{ 
lzhEditorFormat("InsertImage", $.trim($("#imageUrl").valO)); 
$("#uploadLayer").css("display", "none"); 


/可 机 
* 显示 插入 图 片 层 
had 
function lzhEditorAddImage() { 
$("#uploadLayer").css("display", "block"); 
} 


/下 * 
* 更 改编 辑 器 高 度 
a 
function changeEditorArea(type) { 
if(type— 1){ 
document.getElementById("editor").height = parseInt(document 
getElementById("editor").height) +10; 
} else { 
if (parseInt(document.getElementById("editor").height) >=50) { 
document.getElementById("editor").height = parseInt(document 
.getElementById("editor").height) - 10; 


2 
* 更 改 插入 图 片 方式 层 
wl 
function changUploadLayer(x) { 
if(x=0){ 
S$("#upLayerl").css("display", "block'"): 
S$("#upLayer2").css("display", "none"): 
}else{ 
$("#upLayerl").css("display", "none"); 
$("#upLayer2").css("display", "block"); 


} 
sid 
* 将 编辑 器 内 容 赋 给 隐藏 域 (表单 提交 时 调用 ) 

LzhEditor.prototype.Submit = fonctiongO { 

Var content = window.frames["editor"].document.body.outerHTML:; 

content = content 

Teplace( 
<BODY style="PADDING-RIGHT: 5px; PADDING-LEFT: Spx: FONT-SIZE: 


12px; PADDING-BOTTOM: 5px; MARGIN: 0px: PADDING-TOP: 5px">", 
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content = content.replace('</BODY>', "); 
document.getElementById(this.fieldName).value = content; 
上 | 
在 完成 脚本 的 封装 之 后 ， 就 需要 在 网 页 中 载 入 脚本 文件 ， 其 前 提 是 必须 在 网 页 中 载 入 | 1 
jQuery 组 件 ， 否 则 不 能 调用 脚本 文件 中 的 方法 。 | 
接着 ， 编 写 JavaScript 脚本 ， 实 例 化 自 定义 在 线 编辑 器 类 ， 传 递 参数 包括 名 称 、 大 小 、 
脚本 文件 存储 位 置 等 )， 完 成 网 页 中 自 定义 在 线 编辑 器 的 输出 ， 其 关键 代码 如 下 : 
<script language="javascript"> 
var lzhEditor = new LzhEditor('content','500','200',","js/LzhEditor"); 
lzhEditor.Create(); 
</script> | 
最 后 ， 通 过 $_ POST[] 获 取 在 线 编辑 器 中 提交 的 数据 ， 切 记 必须 要 应 用 stripslashes() 函 数 对 | 
在 线 编辑 器 传递 的 数据 进行 解析 ， 否 则 内 容 将 无 法 正常 输出 。 | 
应 用 示例 | 
应 用 自 定 义 在 线 编辑 器 编辑 内 容 ， 并 且 获 取 自 定义 在 线 编辑 器 提交 的 值 ， 其 运行 结果 如 | 
图 16.7 所 示 。 | 


~ 
ee 
[SRseie: Ca A 
ea 
[ 2 旺 妇 二 人 eg- 天 | 
| 日 R: | 九天 各 月 
ED 
HL md EBD lei Ne [| 二 一 


肛 上 九天 丹 彩 : 


E 
站 下 三 洋 帮 攻 。i 


A. | 69 #6 


EE 


图 16.7 自 定义 在 线 编辑 器 应 用 
专家 点 评 
应 用 自 定义 在 线 编辑 器 ， 其 首先 要 在 网 页 中 载 入 jQueryjs 和 LzhEditorjs 文件 ， 然 后 在 网 | 
页 中 正确 地 调用 脚本 中 的 方法 ， 传 递 合适 的 参数 ， 最 后 在 获取 在 线 编辑 器 提交 的 值 时 应 用 | 
stripslashes() 函 数 对 其 进行 解析 。 | 


问题 293 ”如 何 应 用 CKEditor 网 页 编辑 器 ? 


问题 阐述 
论坛 、 博 客 、 微 博 或 在 线 视频 类 网 站 都 具备 发 表 评 论 的 功能 ， 为 了 使 评论 信息 更 加 醒目 ， 
。483 。 Ne 


人 Pp 六 须知 道 的 300 个 问题 


| ED 
| tS 

| 可 以 通过 在 线 编辑 器 对 评论 的 内 容 进行 在 线 编辑 。 那 么 如 何 应 用 在 线 编辑 器 呢 ? 
| 专家 解答 


在 这 里 向 读者 介绍 CKEditor 网 页 编辑 器 的 使 用 方法 。 首 先 下 载 和 安装 CKEditor 编辑 器 。 
CKEditor 是 开源 网 页 编辑 软件 ，FCKEditor 在 2009 年 发 布 更 新 的 3.0 版 本 。CKEditor 是 
”FCKeditor 的 一 个 完全 重 写 版 本 , 它 弥补 了 FCKeditor 打开 速度 慢 的 缺陷 , 并 提供 了 强大 的 、 可 
1 扩展 的 JavaScript API。 


| 说 明 ; 
| 原来 叫 FCK, 是 因为 最 初 的 开发 者 叫 Frederico Calderia Knabben; 现在 叫 CK, 意 指 “Content 
| and Knowledge”. 


CKEditor 的 安装 步骤 非常 简单 ， 如 下 所 示 : 

(1) 从 CKEditor 官方 网 站 上 下 载 CKEditor 的 最 新 版 本 。 

(2) 将 下 载 的 文件 解压 到 项 目 根 目录 下 的 ckeditor 文件 夹 。 
| 安装 成 功 后 ， 可 以 通过 编辑 器 附带 的 一 些 实例 来 验证 其 是 否 能 够 正常 运行 。 附 带 实例 存储 
| 于 “_samples“ 目 录 下 。 访 问 地 址 是 : http://< 网 站 域名 >/<CKEditor 安装 路 径 >/_samples/index.html。 
| CKEditor 安装 成 功 后， 就 需要 在 网 页 中 嵌入 编辑 器 了 。 在 网 页 中 嵌入 CKEditor 编辑 器 的 
| 方法 有 很 多 ， 在 这 里 使 用 PHP 脚本 将 CKEditor 嵌入 到 网 页 中 ， 代 码 如 下 : 


<2php 
include("ckeditor/ckeditor.php"): // 包 含 ckeditor 类 
$CKEditor = new CKEditor0; // 类 的 实例 化 
$CKEditor->basePath = 'ckeditor/'; // 设 置 ckeditor 的 目录 
$CKEditor->config["width'] = 600; /设置 宽 
$CKEditor->textareaAttributes = array("cols" => 20, "rows" => 10); /设置 文本 的 行 、 列 
$CKEditor->replaceAll0; 
2 

应 用 示例 


通过 CKEditor 对 博客 中 发 表 文章 的 内 容 进行 编辑 ， 包 括 对 文字 的 处 理 、 添 加 表格 、 上 传 
| 图 片 和 上 传 Flash 等 。 其 具体 实现 过 程 如 下 : 
| (1) 下 载 、 安 装 CKEditor 编辑 器 。 
| (2) 创建 index.php 文件 ， 创 建 表单 ， 通 过 PHP 脚本 嵌入 CKEditor 编辑 器 ， 将 博客 文章 
| 提交 到 index_ok.php 文件 中 。 其 关键 代码 如 下 : 


<form name="forml" method="post" action="index_ok.php" onSubmit="return chkinput(this)"> 
<input name="topic" type="text" value="" size="40" /> 
<input name="author" type="text" value="" size="30" /> 

| <input name="comefrom" type="text" value="" size="40" /> 

<input type="hidden" name="langver" value="ch" /> 

| <textarea cols="80" id="content" name="content" rows="10"> 请 输入 文章 内 容 :</textarea> 

| <input type="image" name="imageField2" src="images/bg_04.jpg" /> 

| <input type="image" name="imageField" src="images/bg_06.jpg”onClick="form_ reg.reset()" 

1 style="cursor:hand" /> 

| </form> 

| <?php 

| include("ckeditor/ckeditor. php"): /包含 ckeditor 类 

j 
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$CKEditor = new CKEditor(): // 类 的 实例 化 
$CKEditor->basePath = 'ckeditor/'; // 设 置 ckeditor 的 目录 


$CKEditor->config["width'] = 600; /设置 宽 
$CKEditor->textareaAttributes = array("cols" => 20, "rows" => 10); /设置 文本 的 行 、 列 
$CKEditor->replaceAll0; 

?> 


运行 结果 如 图 16.8 所 示 。 
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图 16.8 ”CKEditor 的 应 用 


(3) 在 JavaScript 脚本 中 创建 chkinput0 方 法 ， 验 证 提交 的 数据 是 否 为 空 。 | 
(4) 创建 system 文件 夹 ， 载 入 ADODB 类 库 , 编写 连接 、 操 作 数 据 库 类 的 文件 system.class | 
.inc.php 和 实例 化 类 的 文件 system.inc.php。 | 
(5) 创建 index_ok.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 然 后 通过 数据 库 操作 类 中 的 方法 | 
将 数据 添加 到 数据 表 中 。 | 


专家 点 评 


在 网 页 中 嵌入 CKEditor 网 页 编辑 器 时 ， 一 定 要 设置 正确 的 路 径 ， 如 果 路 径 设置 得 不 正确 ， | 
那么 将 弹出 如 图 16.9 所 示 的 错误 信息 。 | 


该 网 页 上 的 以 后 ， 邓 击 显 示 在 状态 
信 下 的 午 省 图标， 就 可 以 显示 上 述 信息 ， 
厅 同 页 和 会 博 沪 时 航 经 旺 示 该 车 息 扩 ). 
CE we < 
行 : 89 
字符 :1 
错误 :CEEITIOE 未 十 义 
代码 :0 
ee 


ET By 


图 169 路 径 设置 不 正确 导致 的 错误 
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| 之 所 以 会 出 现 上 面 的 错误 ， 是 因为 没有 找到 CKEditor， 其 错误 的 代码 如 下 : 

| <?php 

| include("ckeditor/ckeditor. php"): // 包 含 ckeditor 类 

有 | $CKEditor = new CKEditor0: /类 的 实例 化 
全 | $CKEditor->basePath = '/ckeditor/'; 1/ 设置 ckeditor 的 目录 
| $CKEditor->config["width'] = 600: /设置 宽 

$CKEditor->replaceAll0; 
?> 


| 上 述 问 题 就 出 在 SCKEditor->basePath 的 值 上 ， 如 果 CKEditor 存储 在 根 目录 下 的 ckeditor 
| 文件 夹 下 ， 那 么 SCKEditor->basePath 的 值 应 该 是 “ckeditor ”。 


$CKEditor->textareaAttributes = array("cols" => 20, "rows" => 10); /设置 文本 的 行 、 列 
| 问题 294 如何 利用 WebBrowser 打印 报表 ? 


问题 阐述 
WebBrowser 是 下 内置 的 浏览 器 控件 ， 无 须 用 户 下 载 。 它 的 优点 是 客户 端 独立 完成 打印 目 
标 文档 的 生成 ， 减 轻 服务 器 负荷 ， 缺 点 是 源 文档 的 分 析 操 作 复杂 ， 并 且 要 对 源 文 档 中 要 打印 的 
内 容 进行 约束 。 那 么 ， 如 何 利用 WebBrowser 打印 报表 呢 ? 
专家 解答 
下 内 置 的 WebBrowser 控件 实现 报表 的 打印 等 功能 ， 该 控件 的 具体 参数 如 下 : 
document.all.WebBrowser.Execwb(7,1): 表示 打印 预览 。 
document.all WebBrowser Execwb(6.1): 表示 打印 。 
document.all.WebBrowser.Execwb(6,6): 表示 直接 打印 。 
document.all.WebBrowser.Execwb(8,1): 表示 页 面 设置 。 
document.all.WebBrowser.Execwb(1,1): 打开 页 面 。 
document.all WebBrowserExecwb(2.1): 关闭 所 有 打开 的 下 窗口 。 
document.all.WebBrowser.Execwb(4,1): 保存 网 页 。 
document.all.WebBrowser.Execwb(10,1): 查看 页 面 属性 。 
| document.all WebBrowserExecwb(17.1): 全 选 。 
document.all WebBrowserExecwb(22.1): 刷新 。 
document.all.WebBrowser.Execwb(45,1): 关闭 窗 体 无 提示 。 
通过 HTML 的 object 标签 ， 调 用 WebBrowser 控件 ， 代 码 如 下 : 
<object id="WebBrowser" classid="CISID:8856F961-340A-11D0-A96B-00C04Fd705A2" width="0" 
| height="0"> 
| object> 
| 调用 WebBrowser 控件 的 相应 参数 实现 打印 预览 、 打 印 等 功能 ， 代 码 如 下 : 


<a href—="#" onClick="document.all.WebBrowser.Execwb(7,1)"> 打 印 预览 </a> 
| <a href—="#" onClick="document.all.WebBrowser.Execwb(6,1)"> 打 印 </a> 
| <a href="#" onClick="document.all.WebBrowser. Execwb(6,6)"> 直 接 打 印 </a> 
| <a href="#" onClick="document.all.WebBrowser.Execwb(8,1)"> 页 面 设置 </a> 
J 
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问题 295 ”如何 调用 Word 自动 打印 指定 格式 的 会 议 记 录 ? 
问题 前 述 


在 开发 网 络 应 用 程序 时 ， 有 时 需要 对 输入 的 信息 按 指 定 的 格式 进行 打印 。 例 如 ， 在 办 公 自 
动 化 系统 中 ， 录 入 的 会 议 记录 信息 就 需要 按 指 定 的 格式 打印 。 那 么 ， 如 何 调用 Word 自动 打印 | 
指定 格式 的 会 议 记录 呢 ? | 
专家 解答 
在 PHP 中 利用 Word 自动 打印 指定 格式 的 会 议 记 录 可 以 总 结 为 以 下 几 点 : 
(1) 应 用 JavaScript 的 ActiveXObject0 构 造 函 数 创建 一 个 Word Application 对 象 的 实例 。 | 
(2) 打开 指定 的 Word 文档 ， 这 里 需要 应 用 PHP 的 函数 获取 模板 文档 所 在 路 径 ， 代 码 如 下 : | 
Spath="http://".$HTTP_HOST.dimame($PHP_SELF)."/meetingDot.doc": | 
其 中 ，SHTTP_HOST 变量 用 于 获取 服务 器 名 称 ，dimame(SPHP_SELF) 用 于 获取 路 径 中 的 | 
路 径 名 。 | 
(3) 通 过 Word Application 对 象 的 Bookmarks 集合 的 相应 方法 将 表单 内 容 写 入 指定 的 Word | 
文档 中 。 | 
(4) 调用 wdapp.Application.Printout(): 实 现 自动 打印 Word 文档 。 
应 用 示例 
在 PHP 中 调用 Word 自动 打印 指定 格式 的 会 议 记录 。 其 关键 步骤 如 下 : | 

(1) 创建 一 个 Word 文档 , 在 该 文档 中 设计 好 要 打印 的 会 议 记 录 的 格式 ， 并 将 其 保存 到 示 | 
例 根 目录 下 ， 名 称 为 meetingDot.doc。 

(2) 在 创建 好 的 Word 文档 中 的 指定 位 置 插 入 书签 。 插 入 书签 的 方法 如 下 : 首先 选中 需要 | 
替换 的 文本 ， 然 后 选择 “插入 ”一 “书签 ”命令 ， 在 打开 的 对 话 框 中 输入 书签 名 ， 并 单 击 “ 添 | 
加 ”按钮 即 可 。 | 

(3) 在 示例 主页 面 中 添加 用 于 收集 会 议 信息 的 表单 及 表单 元 素 。 

(4) 应 用 PHP 获取 模板 文档 所 在 的 路 径 ， 代 码 如 下 : 


<2?php 

/获取 模板 文档 所 在 的 路 径 
$path="http://".$HTTP_HOST.dimame($PHP_SELF)."/meetingDot.doc"; 
3 


(5) 编写 自 定义 JavaScript 函数 outDoc()， 用 于 将 表单 收集 的 数据 导出 到 Word， 并 进行 
自动 打印 。 代 码 如 下 : | 


<2php 

/定义 新 型 字符 串 

$str=<<<word 

<script language="javascript"> 

function outDocO{ 
Var wdapp=new ActiveXObject("Word.Application"): 
wdapp.visible=true; 
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wddoc=wdapp.Documents.Open("$path"): // 打 开 指 定 的 文档 
var form=documentallforml; 
title=form .title.value; 
meetingTime=form.meetingTime.value; 
compere=form.compere.value; 
attend=form.attend.value: 
content=form.content.value; 
// 输 出 会 议 标 题 
Tange =wdapp.ActiveDocument.Bookmarks("title").Range; 
Tange.Text=title; 
// 输 出 会 议 时 间 
Tange =wdapp.ActiveDocument.Bookmarks("meetingTime").Range; 
Tange.Text=meetingTime; 
// 输 出 会 议 主持 人 
Tange =wdapp.ActiveDocument.Bookmarks("compere").Range; 
range.Text=compere; 
/输出 出 席 人 员 
Tange =wdapp.ActiveDocument.Bookmarks("attend").Range; 
Tange. Text=attend; 
// 输 出 会 议 内 容 
Tange =wdapp.ActiveDocument.Bookmarks("content").Range; 
range.Text=content; 
wddoc.Application.PrintoutO; 
wdapp=null; 
1 
</script> 
word; 
echo $str; 
> 


(6) 通过 单 击 “Word 打印 ”按钮 调用 自 定 义 JavaScript 函数 outDoc0。 关 键 代码 如 下 : 
<input name="Submit" type="button" class="btn_grey" onClick="outDoc0" value="Word 打印 "> 


运行 本 示例 ， 在 页 面 中 输入 相应 的 会 议 信息 ， 单 击 “Word 打印 ”按钮 ， 即 可 将 录入 的 会 议 
信息 导出 到 指定 的 Word 文档 中 ， 并 自动 按 该 文档 指定 的 格式 打印 。 运 行 结果 如 图 16.10 所 示 。 
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图 16.10 在 PHP 中 调用 Word 自动 打印 指定 格式 的 会 议 记录 
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问题 296 如何 将 Web 页 面 中 的 数据 导出 到 Excel 中 
并 自动 打印 ? 


问题 讲述 

在 Web 网 站 的 实际 项 目 开发 过 程 中 ， 经 常 需要 将 Web 页 面 中 的 数据 导出 到 Excel 中 并 实 

现 自动 打印 。 那 么 ， 如 何 实现 这 一 操作 呢 ? | 

专家 解答 

通过 PrintOut() 方 法 可 实现 自动 打印 Excel 工作 表 中 的 内 容 。 该 方法 用 于 打印 指定 对 象 。 其 | 

语法 格式 如 下 : | 
expression.PrintOut(From, To, Copies, Preview, ActivePrinter, PrintToFile, Collate, PrToFileName) 


参数 说 明 : | 
@ expression: 必 选 参数 。 用 于 返回 “Chart 对 象 "“Charts 集合 对 象 ”“Range 对 象 "“Sheets | 
集合 对 象 人 “Window 对 和 象 ”"”“Workbook 对 象 ””“Worksheet 对 象 ”或 “Worksheets 集合 对 象 ”| 
中 的 某 个 对 象 。 | 
@ From: 可 选 参数 。 用 于 指定 打印 的 开始 页 号 。 如 果 省 略 该 参数 , 将 从 起 始 位 置 开 始 打印 。 | 
@@ To: 可 选 参数 。 用 于 指定 打印 的 终止 页 号 。 如 果 省 略 该 参数 ， 将 打印 至 最 后 一 页 。 | 
@ Copies: 可 选 参数 。 用 于 指定 要 打印 的 份 数 。 如 果 省 略 该 参数 ， 将 只 打印 一 份 。 | 
日 Preview: 可 选 参数 。 值 为 tme 或 false， 如 果 为 tue， 则 Microsoft Excel 打印 指定 对 象 | 
之 前 进行 打印 预览 ， 如 果 为 false 或 者 省 略 此 参数 ， 则 立即 打印 该 对 象 。 | 
@ ActivePrinter: 可 选 参数 。 用 于 设置 活动 打印 机 的 名 称 。 | 
@ PrintToFile: 可 选 参数 。 值 为 true 或 flse， 如 果 为 tue， 则 打印 输出 到 文件 。 如 果 没 有 | 
指定 PrToFileName， 则 Microsoft Excel 将 提示 用 户 输入 要 输出 文件 的 文件 名 。 | 
@ Collate: 可 选 参数 。 当 值 为 tue 时 ， 逐 份 打印 每 份 副本 。 | 
@ PrToFileName: 可 选 参数 。 如 果 将 PrintToFile 设置 为 tue， 则 本 参数 指定 要 打印 到 的 | 
文件 名 。 | 


说 明 : | 
From 参数 和 To 参数 所 描述 的 “页 ” 指 的 是 要 打印 的 页 ， 并 非 指定 工作 表 或 工作 簿 中 的 全 | 
部 页 。 | 


应 用 示例 | 

将 Web 页 面 中 的 话费 列表 导出 到 Excel 中 ， 同 时 实现 保存 并 自动 打印 功能 。 其 具体 实现 步 | 

又 如 下 : | 
(1) 应 用 include 命令 引用 数据 库 配置 文件 来 访问 数据 源 ， 代 码 如 下 : 

<?php include "conn/conn.php"; ?> | 

(2) 应 用 do...while 循环 语句 以 表格 形式 输出 话费 信息 到 浏览 器 中 ， 并 将 显示 话费 信息 | 

的 表格 的 id 设置 为 cost， 因 为 要 将 该 表格 中 的 数据 导出 到 Excel 中 。 关 键 代码 如 下 : | 
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<?php 
! $sql=mysql_query("select * from tb_telcost order by month desc"); 
| $info=mysql fetch array($sql):; 
| dof 
> 
<tr align="center" bgcolor="#FFFFFF"> 
<td width="14%" height="30"><?php echo $info[tel]:?></td> 
<td width="13%"><?php echo $info[bdcost];?></td> 
<td width="14%"><?php echo $info[ipcost];?></td> 
<td width="14%"><?php echo $info[ldcost]:?></td> 
<td width="14%"><?php echo $info[ldcost];?></td> 
<td width="16%"><?php echo $info[total]:?></td> 
<td width="15%"><?php echo $info[month];?></td> 
</tr> 
<?php 
}while($info=mysql fetch array($sq]l)); 
?> 


| (3) 编写 自 定义 JavaScript 函数 outExcel()， 用 于 将 Web 页 面 中 的 话费 列表 信息 和 自 定 义 
| 的 标题 信息 导出 到 Excel 中 ， 并 保存 到 Excel 的 默认 工作 目录 中 ， 同 时 自动 调用 打印 机 进行 打 
印 并 实现 自动 打印 。 代 人 码 如 下 : 


<script language="javascript"> 
function outExcelO{ 
Var table=document.all.cost: 
row=table.rows.length; 
column=table.rows(]).cells.length: 
Var excelapp=new ActiveXObject("Excel.Application"); 


excelapp.visible=true; 

objBook=excelapp. Workbooks.AddO; /添加 新 的 工作 短 
Var objSheet = objBook.ActiveSheet; 

title=objSheetRange("D1") MergeArea: /合并 单元 格 


title.Cells(1,0).Value =paytitle.innerHTML .replace("&nbsp:"," 话 费 单 查询 ");// 输 出 标题 
title.Cells(1,1).Font.Size =16; 
for(i=1;i<row+1;it+){ 
for0=0j<column:j++H){ 
objSheet.Cells(i+1j+1).value=table. rows(i-1).cells().innerHTML .replace("&nbsp;"," 话 费 单 查询 "); 
} 
objBook.SaveAs("telcost.xls"); 
objSheet Printout: /自动 打印 


excelapp.UserControl = true; 


} 
| </script> 


(4) 通过 单 击 “ 打 印 ” 超 链接 调用 自 定义 JavaScript 函数 outExcel0， 代 码 如 下 : 
<a href="#" onClick="outExcel0:"> 打 印 </a> 
运行 本 示例 ， 在 页 面 中 将 显示 话费 列表 ， 单 击 “ 打 印 ” 超 链接 ， 即 可 将 话费 列表 和 自 定义 
的 标题 信息 导出 到 Excel 中 ， 并 保存 到 Excel 的 默认 工作 目录 中 ， 同 时 自动 调用 打印 机 进行 打 
印 。 运 行 结果 如 图 16.11 和 图 16.12 所 示 。 


_ | s。 490 。 


图 16.11 Web 页 面 中 的 数据 显示 
专家 点 评 


在 Excel 中 查看 并 修改 默认 文档 保存 路 径 的 方法 如 下 : 选择 “工具 ”一 “选项 ”命令 ,在 
弹出 的 对 话 框 中 选择 “常规 ”选项 卡 ， 在 该 选项 卡 的 “默认 工作 目录 ”文本 中 将 显示 默认 的 工 | 
作 目 录 ， 读 者 可 以 将 其 修改 为 其 他 可 用 路 径 。 | 


问题 2997 ”如何 将 查询 结果 保存 到 Word 中 ? 


问题 阐述 | 
在 一 些 基于 网 络 的 管理 系统 中 ， 经 常 需要 将 网 页 中 的 数据 保存 到 Word 文档 中 。 那 么 ， 如 | 
何 将 查询 结果 保存 到 Word 中 呢 ? | 


专家 解答 


将 查询 结果 保存 到 Word 文档 中 , 封装 自 定 义 的 Word 类 , 在 Word 类 中 通过 输出 缓冲 函数 | 
将 查询 结果 存储 到 Word 中 。 在 Word 类 中 所 应 用 的 输出 缓冲 函数 包括 : | 
(1) ob_start0: 开始 输出 缓冲 。 此 时 PHP 停止 输出 ， 而 这 以 后 的 输出 都 被 转 到 一 个 内 部 | 
的 缓冲 中 。 | 
(2) ob_get_contents0: 该 函数 返回 内 部 缓冲 的 内 容 。 这 就 等 于 把 这 些 输出 都 变 成 了 字符 串 。 
(3) ob_get_length(): 返回 内 部 缓冲 的 长 度 。 | 

(4) ob_end flushQ0: 结束 输出 缓冲 ， 并 输出 缓冲 中 的 内 容 。 从 这 以 后 的 输出 都 是 正常 输出 。 

(5) ob_end_clean0: 结束 输出 缓冲 ， 并 扔 掉 缓冲 中 的 内 容 。 | 

本 示例 将 数据 库 中 的 数据 保存 到 Word 文档 中 主要 是 通过 自 定义 类 Word 实现 的 。Word 类 | 
的 关键 代码 如 下 : 


function save($path) 
{ 
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$data = ob_get_contents(); 
ob_end clean(); 


S$this->wirtetoword($path, $data); 
| } 
天 六 | function wirtetoword ($fn,$data) 
| { 
| $fp=fopen(Sfn,"wb"); 
WE Sivrite(Sip $data); 

| fclose($fp); 
| } 

} 

?> 


在 该 类 中 定义 了 3 个 成 员 函 数 ， 它 们 实现 的 功能 如 下 : 
(1) start0: 该 成 员 函 数 的 作用 是 定制 要 保存 数据 的 开始 。 
| (2) save(): 该 成 员 函 数 的 作用 是 定制 要 保存 数据 的 结束 ， 同 时 执行 将 数据 库 中 的 数据 保 
| 存 到 Word 中 的 操作 ， 也 就 是 说 ， 所 要 保存 的 数据 必须 限定 在 该 类 的 成 员 函 数 start0 和 成 员 函 
| 数 save0 之 间 。 
| (3) wirtetoword(): 该 成 员 函 数 的 作用 是 实现 将 数据 以 二 进 制 的 形式 保存 到 Word 中 。 
应 用 示例 
将 学 生成 绩 管理 系统 中 查询 出 的 学 生成 绩 数据 存储 到 Word 文档 中 。 其 具体 实现 步骤 如 下 : 
(1) 建立 conn.php 文件 ， 实 现 与 数据 库 的 连接 。 
(2) 建立 word.php 文件 ， 在 该 文件 中 编写 用 于 将 查询 结果 保存 到 Word 中 的 Word 类 。 
(3) 建立 index.php 文件 ， 输 出 查询 结果 ， 并 将 查询 结果 保存 到 Word 中 。 
| 本 示例 的 运行 结果 如 图 16.13 和 图 16.14 所 示 , 其 中 图 16.13 所 示 为 所 查询 到 的 学 生成 绩 的 
| 相关 信息 ， 图 16.14 所 示 为 将 查询 结果 保存 到 Word 中 的 结果 。 
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图 16.13 查询 结果 图 16.14 将 查询 结果 保存 到 Word 中 


问题 298” 如何 将 查询 结果 保存 到 上 Excel 中 ? 


| 问题 阐述 


| Excel 在 办 公 系 统 中 被 广泛 应 用 ， 利 用 该 办 公 软 件 可 以 保存 大 量 的 统计 信息 ， 以 方便 相关 
| 工作 人 员 的 打印 维护 等 操作 ， 而 大 多 数 管理 系统 的 数据 信息 都 保存 到 数据 库 中 。 那 么 如 何 将 数 
| 据 库 中 的 信息 保存 到 Excel 表格 中 呢 ? 
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专家 解答 


将 数据 库 中 的 信息 保存 到 Excel 表格 中 ， 其 关键 是 自 定义 的 excel 类 的 封装 。 在 该 类 中 定 | 
义 3 个 成 员 函 数 ， 其 中 start0 用 于 限定 要 保存 的 数据 的 开始 ; save0 用 于 限定 要 保存 的 数据 的 结 | f 
束 ， 并 调用 成 员 函 数 wirtetoexcel0 实 现 将 查询 结果 以 二 进 制 的 形式 保存 到 Excel 表格 中 。excel | 图 思 
类 的 语法 如 下 : Es 
MNote 
<2php | 
class excel{ | 
function startO{ 
ob_start(); 
} 
function save($path){ 
$data = ob_get_contents(); 
ob_end clean(); 
S$this->wirtetoexcel($path,$data); 


function wirtetoexcel ($fn,$data){ 
$fp=fopen($fn,"wb"); 
fwrite($fp,$data); 
felose($fp); 
} 
} 


?> 
应 用 示例 
将 从 数据 库 中 查询 出 的 数据 保存 到 Excel 表格 中 。 其 具体 实现 步骤 如 下 : 
(1) 建立 数据 连接 文件 conn.php 实现 与 数据 库 的 连接 。 
(2) 封装 excel 类 。 
(3) 创建 ndex.php 文件 ， 实 现 将 查询 结果 保存 到 数据 库 中 。 | 
运行 本 示例 ， 在 图 16.15 所 示 的 表单 中 输入 要 查询 的 学 生 的 学 号 ， 单 击 “ 将 表格 内 容 保存 | 
到 Excel” 按 钮 ， 将 查询 结果 保存 到 Excel 表格 中 ， 如 图 16.16 所 示 。 | 


BW crosoft ixcel — date 21s 0 
国 ] 文件 人 护 儿 加 视 目 扩 插入 区 格式 @) 工具 四 数据 Q) 窗口 
人 百 记 络 果 页 面 MRsoft WD -日 
[ | em [| 革 。 
AL 


-Ll -B17 JL sla :HA 曙 
宁 


HE 


图 16.15 查询 结果 图 16.16 将 查询 结果 保存 到 Excel 中 的 结果 
问题 299 如何 实现 在 线 支 付 (工商 银行 ) ? 
问题 阐述 


随 着 Internet 的 不 断 发 展 ， 电 子 商 务 已 经 成 为 21 世纪 网 络 发 展 的 主流 ， 网 上 购物 和 网 上 支 
.493. a 


| 专家 解答 


bg 1. 解决 思路 
针 - 要 在 电子 商务 网 站 中 通过 工商 银行 完成 在 线 支付 的 操作 , 其 具体 的 开发 流程 如 图 16.17 所 示 。 
St | 
| 订单 查询 二 
NE 中 指定 商品 
| 刘 人 人 | mb 证 | we| | 清和 9 车 | | | 请 的 交 
用 查看 网 和 车 内 商品 下 外 续 购物 
商品 全 种 小 计 p 
万 
商品 全 家 总 计 节 
工行 支 付 订单 打印 
| 提亲 日 | 
取消 订单 订单 预 


图 16.17 在 线 支 付 的 操作 流程 


| 从 图 16.17 中 可 以 看 出 ,工商 银行 在 线 支 付 是 整个 电子 商务 流程 中 最 重要 的 一 个 操作 环节 ， 
| 在 执行 该 操作 之 前 ， 必 须 逐 一 完成 前 面 的 操作 环节 ， 和 否则 单独 执行 工商 银行 在 线 支付 没有 任何 
| 意义 。 所 以 ， 在 开发 本 示例 时 ， 不 只 要 完成 工商 银行 在 线 支付 功能 的 开发 ， 同 时 还 要 进行 商品 
| 浏览 、 购 物 车 、 生 成 订单 和 提交 订单 等 功能 的 开发 。 
2.， 具体 方法 
| 工商 银行 在 线 支付 根据 工商 银行 提供 的 在 线 支付 接口 , 将 订单 中 的 数据 以 表单 的 形式 提交 到 
| 工商 银行 指定 的 网 址 中 。 具 体 详 细 的 参数 设置 可 以 参考 工商 银行 提供 的 文档 说 明 书 和 示例 程序 。 
| 在 实现 与 工商 银行 接口 的 连接 之 前 ， 首 先 要 从 工商 银行 提供 的 说 明 书 中 找到 下 面 两 个 文件 : 
(1) 开发 API 接口 \LIB\windows\win32\infosecapi.dll (Windows XP 系统 使 用 的 文件 )。 
(2) 开发 API 接口 ASPICBCEBankUtil.dll。 
| 将 其 复制 到 系统 文件 夹 下 的 \WINDOWS\system32 下 ， 在 运行 命令 中 输入 “regsvr32 
| ICBCEBankUtildl”， 加 载 服务 。 
| ”通过 PHP 实现 网 站 与 工商 银行 接口 交互 的 方法 如 下 : 
| 首先 ， 通 过 com 类 中 的 init0 方 法 对 订单 签名 数据 和 商城 的 公 钥 进行 加 密 处 理 ， 并 且 将 指 
| 定 的 数据 以 表单 的 形式 提交 到 工行 指定 的 页 面 中 。 其 关键 代码 如 下 : 
| $src="ICBC_ PERBANK B2C1.0.0.0".$infomer["merid"].$infomer["meracct"].$returnaddress."HS" 
| .$ddnumber.$amount."0010".$nowtime."0"; 


$com=new com(ICBCEBANKUTIL.B2CUtil): 
S$re=$com->init("c:\WINDOWS\user.crt","d: \user.crt","d: \user.key","00"); 


$ssrc=$com->signC($src, strlen($sre)); /订单 签名 数据 
$rc=$com->verifySignC($src,strlen($src), $ssrc, strlen($ssrc)): 
$cert=$com->getCert(1): // 商 城 证 书 公 钥 


| 完成 订单 签名 数据 和 商城 证 书 公 钥 的 加 密 处 理 ， 这 里 将 商户 的 测试 证 书 存储 到 CNWINDOWS 
， 目 录 下 ， 设 置 商 户 的 私 钥 口 令 是 00。 
然后 ， 创 建 form 表单 ， 将 指定 的 数据 提交 到 工行 指定 的 网 站 中 。 关 键 代码 如 下 : 
了 .494 。 
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<form action="https://mybank.icbc.com.cn " method="post" name="form bank"> 
<input name="interfaceName" type="hidden" value="ICBC PERBANK B2C"/> 
<input name="interfaceVersion" type="hidden" value="1.0.0.0"/> 
<input name="orderid" type="hidden" value="<{$ddnumber}>"/> 
<input name="amount" type="hidden" value="<{$amount}>"/> 
<input name="curType" type="hidden" value="001"/> 
<input name="merID" type="hidden" value="<{$merid}>"/> 
<input name="merAcct" type="hidden" value="<{$meracct}>"/> 
<input name="verifyJoinFlag" type="hidden" value="0"/> 
<input name="notifyType" type="hidden" value="HS"/> 
<input name="merURL" type="hidden" value="<{$retumaddress}>"/> 
<input name="resultType" type="hidden" value="0"/> 
<input name="orderDate" type="hidden" value="<{$nowtime}>"/> 
<input name="merSignMsg" type="hidden" value="<{$ssre}>" /> 
<input name="merCert" type="hidden" value="<{$cert}>" /> 
</form> 


3， 应 用 示例 
吉林 省 明日 科技 有 限 公 司 的 编程 词典 网 (http://www.mrbccd.com) 就 是 一 个 电子 商务 网 站 ， 
它 完美 地 实现 了 编程 词典 的 在 线 购买 功能 , 并 且 为 用 户 提供 了 多 种 支付 方式 ,如 图 16.18 所 示 。 
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图 16.18 编程 词典 网 站 的 支付 方式 
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在 本 示例 中 , 将 对 编程 词典 网 站 中 的 工商 银行 在 线 支付 功能 进行 剖析 .其 具体 实现 步骤 如 下 : 
(1) 本 示例 应 用 Smarty 进行 开发 ， 通 过 ADODB 操作 MySQL 数据 库 ， 并 将 Smarty 模板 
和 ADODB 类 库存 储 在 system 文件 夹 下 。 同 时 将 连接 、 操 作 MySQL 的 方法 以 及 Smarty 的 配 
置 方法 都 定义 到 类 文件 中 ， 并 存储 在 system 文件 夹 下 。system 文件 夹 的 架构 如 图 16.19 所 示 。 
(2) 创建 动态 PHP 文件 index.php 和 模板 文件 index.phtml， 实 现 商 品 的 分 页 、 分 类 浏览 ， 
并 且 创 建 购买 和 查看 购物 车 的 超 链 接 。 
在 index.php 中 ， 调 用 包含 文件 ， 完 成 数据 库 的 连接 ， 应 用 数据 库 管理 类 AdminDBO 中 的 


| ExecSQL() 方 法 读 取 数据 中 的 数据 ， 并 将 查询 结果 赋 给 模板 变量 。 


应 用 分 页 类 SepPage0 中 的 ShowDate( 方 法 ， 执 行 分 页 查询 ， 实 现 分 页 功能 ， 将 分 页 结果 


赋 给 模板 变量 ， 应 用 分 页 类 SepPage0 中 的 ShowPage() 方 法 执行 分 页 操作 ， 完 成 分 页 超 链接 的 
| 创建 ， 并 将 返回 结果 赋 给 模板 变量 。 最 终 指定 模板 页 index.phtml。 


在 index.phtml 模板 页 中 ， 根 据 PHP 模板 变量 传递 的 值 ， 应 用 section 循环 语句 完成 商品 信 


息 的 分 页 输出 ， 同 时 创建 购买 商品 和 查看 购物 车 的 超 链 接 ， 传 递 的 参数 为 商品 的 ID。 


(3) 创建 购物 车 。 在 购物 车 功能 模块 中 包含 更 改 商品 数量 、 删 除 商品 、 商 品 小 计 、 商 品 


总 计 、 清 空 购物 车 和 结算 等 。 购 物 车 的 操作 页 面 如 图 16.20 所 示 。 
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(4) 创建 、 生 成 订单 ， 完 成 订单 打印 和 打印 预览 功能 的 实现 。 创 建 订单 应 用 的 是 shopping_ 


| order form.php 和 shopping_order form.phtml 文件 ， 订 单 处 理应 用 的 是 shopping_order form_ 
| okphp 文件 。 


生成 订单 通过 create_order_form.php 和 create_order_form.phtml 文件 来 完成 。 
订单 打印 和 打印 预览 功能 应 用 的 是 printwindowphp 和 printwindow.phtml 文件 。 首 先 在 


| create_order_form.phtml 文件 中 应 用 openprintwindow0 函 数 将 参数 传递 到 printwindow.php 文件 


中 ， 然 后 在 该 文件 中 完成 订单 打印 和 预览 的 操作 。 
(5) 创建 submit_order form.php 和 submit_order form.phtml 文件 ， 实 现 订单 处 理 。 

在 submit_order_form.php 中 根据 订单 号 查询 出 订单 的 详细 信息 , 并 将 查询 结果 赋 给 指定 的 
模板 变量 ， 实现 网 站 与 工商 银行 接口 的 连接 ， 定 义 模板 变量 将 经 过 加 密 的 数据 传递 到 模板 页 的 
表单 中 ， 通 过 form 表单 向 工商 银行 指定 的 网 站 中 传递 数据 。 其 关键 代码 如 下 : 

/组 织 要 签名 的 数据 串 
$src="ICBC_ PERBANK B2C]1.0.0.0".$infomer["merid"].$infomer["meracct"].$returnaddress."HS" 
.$ddnumber.$amount."0010".$nowtime."0"; 


$com=new com(ICBCEBANKUTIL.B2CUtil): 
S$re=$com->init("c:\WINDOWS\user.crt","c:\WINDOWS\user.crt","c:\WINDOWS\user.key","00"); 


$ssrc=$com->signC($src, strlen($src)): // 订 单 签名 数据 
$rc=$com->verifySignC($src,strlen($src), $ssrc, strlen($ssrc)): 

S$cert=$com->getCert(1); // 商 城 证 书 公 钥 

$smarty->assign("ssre", $ssrc); // 定 义 模板 变量 传递 加 密 的 订单 签名 数据 
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S$smarty->assign("cert",$cert); // 定 义 模板 变量 传递 加 密 的 商城 公 钥 
$nowtime=date("YmdHis"); 

/将 指定 的 数据 通过 模板 变量 传递 到 表单 中 

$sqlmer=mysql_query("select merid,meracct from tb_system where mark=1",$conn); 
Sinfomer=mysql fetch array($sqlmer): 
$returnaddress="http://www.cccxy.com/getinfo.asp"; 
$smarty->assign("ddnumber",$ddnumber); 

$smarty->assign("nowtime", $nowtime); 

$smarty->assiegn("returnaddress", $returnaddress); 
$smarty->assign("merid",$infomer["merid"]); 

$smarty->assign("meracct", $infomer["meracct"]); 

$smarty->assign("sre", $sre); 


在 submit_order_form.phtml 文件 中 创建 form 表单 ,将 从 模板 变量 中 获取 的 数据 通过 表单 提 | 
交 到 工商 银行 指定 的 网 站 中 。 其 关键 代码 如 下 : 


<!-- 工 行 支付 的 接口 操作 ， 提 交 的 数据 --> 
<form action="https://mybank.icbe.com.cn/" method="post" name="form bank"> 
<input name="interfaceName" type="hidden" value="ICBC_ PERBANK B2C"/> 
<input name="interfaceVersion" type="hidden" value="1.0.0.0"/> 
<input name="orderid" type="hidden" value="<{$ddnumber}>"/> 
<input name="amount" type="hidden" value="<{$amount}>"/> 
<input name="curType" type="hidden" value="001"/> 
<input name="merID" type="hidden" value="<{$merid}>"/> 
<input name="merAcct" type="hidden" value="<{$meracct}>"/> 
<input name="verifyJoinFlag" type="hidden" value="0"/> 
<input name="notifyType" type="hidden" value="HS"/> 
<input name="merURL" type="hidden" value="< {$returnaddress}>"/> 
<input name="resultType" type="hidden" value="0"/> 
<input name="orderDate" type="hidden" value="<{$nowtime}>"/> 
<input name="merSignMsg" type="hidden" value="<{$ssre}>" /> 
<input name="merCert" type="hidden" value="<{$cert}>" /> 
</form> | 
<td width="124"><img src="images/sy_18.jpg" width="70" height="25" onClick="javascript:form | 
bank.submitO:" style="cursor:hand"/></td> | 
<!- > 


这 里 给 出 编程 词典 网 站 通过 工商 银行 进行 在 线 支付 的 操作 页 面 ， 如 图 16.21 所 示 。 
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到 Pp 六 须 知道 的 300 个 辣 是 


专家 点 评 


这 里 讲解 的 是 在 通过 PHP 语言 开发 的 电子 商务 网 站 中 如 何 与 工商 银行 进行 交互 的 方法 ， 
其 中 涉及 工商 银行 提供 的 参数 和 设置 都 是 个 人 虚拟 的 , 真实 的 参数 将 在 实际 的 运作 中 由 工商 银 
行 提供 。 当 实际 需要 使 用 此 项 功能 时 ， 需 向 工商 银行 进行 申请 ， 工 商 银行 会 提供 相应 的 参数 。 


问题 300 ”如 何 实现 在 线 支 付 ( 支 付 宝 支 付 )? 


问题 阐述 


通过 工商 银行 可 以 实现 在 线 支付 的 操作 ， 而 通过 支付 宝 也 可 以 实现 在 线 支付 的 操作 。 下 面 
我 们 就 来 探讨 一 下 如 何 通 过 支付 宝 实现 在 线 支付 。 


专家 解答 


使 用 支付 宝 进行 在 线 支付 ， 必 须 先 在 支付 宝 中 注册 一 个 支付 宝 账 户 ， 然 后 下 载 支付 宝 提 供 
的 接口 文件 ， 通 过 该 接口 文件 可 实现 电子 商务 网 站 与 支付 
宝 的 交互 。 
在 电子 商务 网 站 中 需要 嵌入 几 个 支付 宝 提供 的 文件 ， 
如 图 16.22 所 示 。 a 
在 支付 宝 提供 的 接口 文件 中 提供 了 完整 的 示例 程序 ， 四 下 全 
只 要 依照 示例 程序 中 的 操作 方法 ， 将 电子 商务 网 站 中 指定 人 
的 数据 提交 到 支付 宝 提供 的 数组 参数 中 ,支付宝 即 会 通过 自己 定义 的 方法 将 数据 提交 到 指定 的 
网 站 中 。 实 现 网 站 与 支付 宝 交互 的 关键 代码 如 下 : 
require_once("alipay_service.php"); // 调 用 支付 宝 提 供 的 文件 
require_once("alipay_config.php"): /调用 支付 宝 提供 的 文件 
$array=explode("@",$info["spe"]); 
$arraynum=explode("(@",$info["sle"]); 
Sarrayinfo=array(); // 创 建 数 组 
S$arrayinfo_count=array(); // 创 建 数 组 
for($i=0:;$i<count($array):$i++){ 
if($aray[$i]!="){ 
Sm=$itl; 
S$sqlcart=mysql_query("select * from tb_commodity where tb_commodity_id=".$array[$i]."™", 


支付 宝 提供 的 文件 
支付 宝 提 殿 的 文件 
支付 宝 提供 的 文件 
支付 宝 提供 的 文件 


六 
全 
年 
和 
be 盐 付 宝 提 供 的 文件 


S$conn); 
S$infocart=mysql fetch array($sqlcart): / 读 取 数 据 库 中 数据 
array_push($arrayinfo,$infocart[tb_commodity_name]):// 将 购物 商品 的 名 称 写 入 数组 中 
array_push($arrayinfo count" 商品 $m : $infocart[tb commodity name] 数量 : 
Sarraynum[$i]"); // 将 购物 商品 的 数量 写 入 数组 中 
} 
1 
$commodity_name=implode(',.$arrayinfo): /获取 商品 名 称 
$commodity_count=implode(',",.$arrayinfo_count); 1/ 获取 商品 描述 信息 


$parameter = array( 
"service" => "trade_create_ by_buyer"， // 交 易 类 型 ， 必 填 实物 交易 =trade_create_by_buyer 
"partner" =>$partner, /合作 商户 号 
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"Teturn_url" =>$return_url, // 同 步 返回 

"notify_url" =>$notify_url, // 异 步 返回 

"_input_charset" => $_input_charset, // 字 符 集 ， 默 认为 GBK 

"subject" => $commodity_name, // 商 品名 称 ， 必 填 

"body" => $commodity_count, // 商 品 描述 ， 必 填 

"out trade no" => $ddnumber, /商品 外 部 交易 号 ， 必 填 ， 每 次 测试 都 必须 修改 

"logistics fee"=>$yprice, // 物 流 配 送 费 用 

"logistics_ payment"=>'BUYER_PAY"， // 物 流 配送 费用 付款 方式 ;BUYER_PAY ( 买 家 支付 ) 
"logistics_ type"=>'EXPRESS', /物流 配送 方式 : EMS(EMS)、EXPRESS (其 他 快递 ) | 
"price" => $amount, // 商 品 单价 ， 必 填 | 
"payment type"=>"1", // 默 认为 1， 不 需要 修改 | 
"quantity" => "1", // 商 品 数量 ， 必 填 | 
"show_url" => $show_url, // 商 品 相 关 网 站 | 
"seller_ email" => $seller email // 卖 家 邮箱 ， 必 填 | 
这 | 


$alipay = new alipay_service($parameter$security_code,$sign_type): 
$link=$alipay->create_url0: 
$smarty->assign("link", $link); | 


这 里 讲解 的 是 通过 PHP 来 实现 电子 商务 网 站 与 支付 宝 进行 交互 的 方法 ， 其 中 涉及 支付 宝 | 
提供 的 参数 和 设置 都 是 虚拟 的 ， 真 实 的 参数 将 在 实际 的 运作 中 由 支付 宝 提供 。 
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